require 'unit_spec_helper'

describe Shoulda::Matchers::ActiveRecord::AssociationMatcher, type: :model do
  include UnitTests::ApplicationConfigurationHelpers

  context 'belong_to' do
    it 'accepts a good association with the default foreign key' do
      expect(belonging_to_parent).to belong_to(:parent)
    end

    it 'rejects a nonexistent association' do
      expect(define_model(:child).new).not_to belong_to(:parent)
    end

    it 'rejects an association of the wrong type' do
      define_model :parent, child_id: :integer
      expect(define_model(:child) { has_one :parent }.new).not_to belong_to(:parent)
    end

    it 'rejects an association that has a nonexistent foreign key' do
      define_model :parent
      expect(define_model(:child) { belongs_to :parent }.new).not_to belong_to(:parent)
    end

    it 'accepts an association with an existing custom foreign key' do
      define_model :parent
      define_model :child, guardian_id: :integer do
        belongs_to :parent, foreign_key: 'guardian_id'
      end

      expect(Child.new).to belong_to(:parent)
    end

    it 'accepts an association with an existing custom foreign key and type' do
      define_model :parent
      define_model :child, ancestor_id: :integer, ancestor_type: :string do
        belongs_to :parent, polymorphic: true, foreign_key: 'ancestor_id', foreign_type: 'ancestor_type'
      end

      expect(Child.new).to belong_to(:parent).
        with_foreign_key(:ancestor_id).
        with_foreign_type(:ancestor_type)
    end

    it 'accepts an association using an existing custom primary key' do
      define_model :parent
      define_model :child, parent_id: :integer, custom_primary_key: :integer do
        belongs_to :parent, primary_key: :custom_primary_key
      end
      expect(Child.new).to belong_to(:parent).with_primary_key(:custom_primary_key)
    end

    it 'rejects an association with a bad :primary_key option' do
      matcher = belong_to(:parent).with_primary_key(:custom_primary_key)

      expect(belonging_to_parent).not_to matcher

      expect(matcher.failure_message).to match(/Child does not have a custom_primary_key primary key/)
    end

    if rails_version >= 7.1 && !rails_gt_8_0?
      it 'accepts an association using an valid :query_constraints option' do
        define_model :parent, name: :string
        define_model :child, parent_id: :integer, parent_name: :string do
          belongs_to :parent, query_constraints: [:parent_id, :parent_name]
        end

        expect(Child.new).to belong_to(:parent).with_query_constraints([:parent_id, :parent_name])
      end

      it 'rejects an association with a bad :query_constraints option' do
        matcher = belong_to(:parent).with_query_constraints([:parent_id, :parent_name])

        expect(belonging_to_parent).not_to matcher

        expect(matcher.failure_message).to match(/Child should have :query_constraints options set to \[:parent_id, :parent_name\]/)
      end
    end

    it 'accepts a polymorphic association' do
      define_model :child, parent_type: :string, parent_id: :integer do
        belongs_to :parent, polymorphic: true
      end

      expect(Child.new).to belong_to(:parent)
    end

    it 'accepts an association with a valid :dependent option' do
      expect(belonging_to_parent(dependent: :destroy)).
        to belong_to(:parent).dependent(:destroy)
    end

    it 'rejects an association with a bad :dependent option' do
      expect(belonging_to_parent).not_to belong_to(:parent).dependent(:destroy)
    end

    it 'accepts an association with a valid :counter_cache option' do
      expect(belonging_to_parent(counter_cache: :attribute_count)).
        to belong_to(:parent).counter_cache(:attribute_count)
    end

    it 'defaults :counter_cache to true' do
      expect(belonging_to_parent(counter_cache: true)).
        to belong_to(:parent).counter_cache
    end

    it 'rejects an association with a bad :counter_cache option' do
      expect(belonging_to_parent(counter_cache: :attribute_count)).
        not_to belong_to(:parent).counter_cache(true)
    end

    it 'rejects an association that has no :counter_cache option' do
      expect(belonging_to_parent).not_to belong_to(:parent).counter_cache
    end

    if rails_version >= 7.2
      it 'accepts :counter_cache with a hash' do
        expect(belonging_to_parent(counter_cache: { active: true })).
          to belong_to(:parent).counter_cache
      end

      it 'accepts :counter_cache with active false when passed' do
        expect(belonging_to_parent(counter_cache: { active: false })).
          to belong_to(:parent).counter_cache(active: false)
      end

      it 'rejects :counter_cache with active false when mismatch' do
        expect(belonging_to_parent(counter_cache: { active: true })).
          not_to belong_to(:parent).counter_cache(active: false)
      end

      it 'rejects :counter_cache with when column mismatch' do
        expect(belonging_to_parent(counter_cache: { column: :attribute_count })).
          not_to belong_to(:parent).counter_cache(true)
      end
    end

    it 'accepts an association with a valid :inverse_of option' do
      expect(belonging_to_with_inverse(:parent, :children)).
        to belong_to(:parent).inverse_of(:children)
    end

    it 'rejects an association with a bad :inverse_of option' do
      expect(belonging_to_with_inverse(:parent, :other_children)).
        not_to belong_to(:parent).inverse_of(:children)
    end

    it 'rejects an association that has no :inverse_of option' do
      expect(belonging_to_parent).
        not_to belong_to(:parent).inverse_of(:children)
    end

    it 'accepts an association with a valid :conditions option' do
      define_model :parent, adopter: :boolean
      define_model(:child, parent_id: :integer).tap do |model|
        define_association_with_conditions(model, :belongs_to, :parent, adopter: true)
      end

      expect(Child.new).to belong_to(:parent).conditions(adopter: true)
    end

    it 'rejects an association with a bad :conditions option' do
      define_model :parent, adopter: :boolean
      define_model :child, parent_id: :integer do
        belongs_to :parent
      end

      expect(Child.new).not_to belong_to(:parent).conditions(adopter: true)
    end

    it 'accepts an association without a :class_name option' do
      expect(belonging_to_parent).to belong_to(:parent).class_name('Parent')
    end

    it 'accepts an association with a valid :class_name option' do
      define_model :tree_parent
      define_model :child, parent_id: :integer do
        belongs_to :parent, class_name: 'TreeParent'
      end

      expect(Child.new).to belong_to(:parent).class_name('TreeParent')
    end

    it 'rejects an association with a bad :class_name option' do
      expect(belonging_to_parent).not_to belong_to(:parent).class_name('TreeChild')
    end

    it 'rejects an association with non-existent implicit class name' do
      expect(belonging_to_non_existent_class(:child, :parent)).not_to belong_to(:parent)
    end

    it 'rejects an association with non-existent explicit class name' do
      expect(belonging_to_non_existent_class(:child, :parent, class_name: 'Parent')).not_to belong_to(:parent)
    end

    it 'adds error message when rejecting an association with non-existent class' do
      message = 'Expected Child to have a belongs_to association called parent (Parent2 does not exist)'
      expect {
        expect(belonging_to_non_existent_class(:child, :parent, class_name: 'Parent2')).to belong_to(:parent)
      }.to fail_with_message(message)
    end

    it 'accepts an association with a namespaced class name' do
      define_module 'Models'
      define_model 'Models::Organization'
      user_model = define_model 'Models::User', organization_id: :integer do
        belongs_to :organization, class_name: 'Organization'
      end

      expect(user_model.new).
        to belong_to(:organization).
        class_name('Organization')
    end

    it 'resolves class_name within the context of the namespace before the global namespace' do
      define_module 'Models'
      define_model 'Organization'
      define_model 'Models::Organization'
      user_model = define_model 'Models::User', organization_id: :integer do
        belongs_to :organization, class_name: 'Organization'
      end

      expect(user_model.new).
        to belong_to(:organization).
        class_name('Organization')
    end

    it 'accepts an association with a matching :autosave option' do
      define_model :parent, adopter: :boolean
      define_model :child, parent_id: :integer do
        belongs_to :parent, autosave: true
      end
      expect(Child.new).to belong_to(:parent).autosave(true)
    end

    it 'rejects an association with a non-matching :autosave option with the correct message' do
      define_model :parent, adopter: :boolean
      define_model :child, parent_id: :integer do
        belongs_to :parent, autosave: false
      end

      message = 'Expected Child to have a belongs_to association called parent (parent should have autosave set to true)'
      expect {
        expect(Child.new).to belong_to(:parent).autosave(true)
      }.to fail_with_message(message)
    end

    context 'an association with a :validate option' do
      [false, true].each do |validate_value|
        context "when the model has validate: #{validate_value}" do
          it 'accepts a matching validate option' do
            expect(belonging_to_parent(validate: validate_value)).
              to belong_to(:parent).validate(validate_value)
          end

          it 'rejects a non-matching validate option' do
            expect(belonging_to_parent(validate: validate_value)).
              not_to belong_to(:parent).validate(!validate_value)
          end

          it 'defaults to validate(true)' do
            if validate_value
              expect(belonging_to_parent(validate: validate_value)).
                to belong_to(:parent).validate
            else
              expect(belonging_to_parent(validate: validate_value)).
                not_to belong_to(:parent).validate
            end
          end

          it 'will not break matcher when validate option is unspecified' do
            expect(belonging_to_parent(validate: validate_value)).to belong_to(:parent)
          end
        end
      end
    end

    context 'an association without a :validate option' do
      it 'accepts validate(false)' do
        expect(belonging_to_parent).to belong_to(:parent).validate(false)
      end

      it 'rejects validate(true)' do
        expect(belonging_to_parent).not_to belong_to(:parent).validate(true)
      end

      it 'rejects validate()' do
        expect(belonging_to_parent).not_to belong_to(:parent).validate
      end
    end

    context 'an association with a :touch option' do
      [false, true].each do |touch_value|
        context "when the model has touch: #{touch_value}" do
          it 'accepts a matching touch option' do
            expect(belonging_to_parent(touch: touch_value)).
              to belong_to(:parent).touch(touch_value)
          end

          it 'rejects a non-matching touch option' do
            expect(belonging_to_parent(touch: touch_value)).
              not_to belong_to(:parent).touch(!touch_value)
          end

          it 'defaults to touch(true)' do
            if touch_value
              expect(belonging_to_parent(touch: touch_value)).
                to belong_to(:parent).touch
            else
              expect(belonging_to_parent(touch: touch_value)).
                not_to belong_to(:parent).touch
            end
          end

          it 'will not break matcher when touch option is unspecified' do
            expect(belonging_to_parent(touch: touch_value)).to belong_to(:parent)
          end
        end
      end
    end

    context 'an association without a :touch option' do
      it 'accepts touch(false)' do
        expect(belonging_to_parent).to belong_to(:parent).touch(false)
      end

      it 'rejects touch(true)' do
        expect(belonging_to_parent).not_to belong_to(:parent).touch(true)
      end

      it 'rejects touch()' do
        expect(belonging_to_parent).not_to belong_to(:parent).touch
      end
    end

    context 'given the association is neither configured to be required nor optional' do
      context 'when qualified with required(true)' do
        context 'when belongs_to is configured to be required by default' do
          it 'passes' do
            with_belongs_to_as_required_by_default do
              expect(belonging_to_parent).to belong_to(:parent).required(true)
            end
          end
        end

        context 'when belongs_to is not configured to be required by default' do
          it 'fails with an appropriate message' do
            with_belongs_to_as_optional_by_default do
              assertion = lambda do
                expect(belonging_to_parent).
                  to belong_to(:parent).required(true)
              end

              message = format_message(<<-MESSAGE, one_line: true)
                Expected Child to have a belongs_to association called parent
                (and for the record to fail validation if :parent is unset;
                i.e., either the association should have been defined with
                `required: true`, or there should be a presence validation on
                :parent)
              MESSAGE

              expect(&assertion).to fail_with_message(message)
            end
          end
        end
      end

      context 'when qualified with required(false)' do
        context 'when belongs_to is configured to be required by default' do
          it 'fails with an appropriate message' do
            with_belongs_to_as_required_by_default do
              assertion = lambda do
                expect(belonging_to_parent).
                  to belong_to(:parent).required(false)
              end

              message = format_message(<<-MESSAGE, one_line: true)
                Expected Child to have a belongs_to association called parent
                (and for the record not to fail validation if :parent is
                unset; i.e., either the association should have been defined
                with `required: false`, or there should not be a presence
                validation on :parent)
              MESSAGE

              expect(&assertion).to fail_with_message(message)
            end
          end
        end

        context 'when belongs_to is not configured to be required by default' do
          it 'passes' do
            with_belongs_to_as_optional_by_default do
              expect(belonging_to_parent).to belong_to(:parent).required(false)
            end
          end
        end
      end

      context 'when qualified with optional(true)' do
        context 'when belongs_to is configured to be required by default' do
          it 'fails with an appropriate message' do
            with_belongs_to_as_required_by_default do
              assertion = lambda do
                expect(belonging_to_parent).
                  to belong_to(:parent).optional(true)
              end

              message = format_message(<<-MESSAGE, one_line: true)
                Expected Child to have a belongs_to association called parent
                (and for the record not to fail validation if :parent is
                unset; i.e., either the association should have been defined
                with `optional: true`, or there should not be a presence
                validation on :parent)
              MESSAGE

              expect(&assertion).to fail_with_message(message)
            end
          end
        end

        context 'when belongs_to is not configured to be required by default' do
          it 'passes' do
            with_belongs_to_as_optional_by_default do
              expect(belonging_to_parent).to belong_to(:parent).optional(true)
            end
          end
        end
      end

      context 'when qualified with optional(false)' do
        context 'when belongs_to is configured to be required by default' do
          it 'passes' do
            with_belongs_to_as_required_by_default do
              expect(belonging_to_parent).to belong_to(:parent).optional(false)
            end
          end
        end

        context 'when belongs_to is not configured to be required by default' do
          it 'fails with an appropriate message' do
            with_belongs_to_as_optional_by_default do
              assertion = lambda do
                expect(belonging_to_parent).
                  to belong_to(:parent).optional(false)
              end

              message = format_message(<<-MESSAGE, one_line: true)
                Expected Child to have a belongs_to association called parent
                (and for the record to fail validation if :parent is
                unset; i.e., either the association should have been defined
                with `optional: false`, or there should be a presence
                validation on :parent)
              MESSAGE

              expect(&assertion).to fail_with_message(message)
            end
          end
        end
      end

      context 'when qualified with nothing' do
        context 'when belongs_to is configured to be required by default' do
          it 'passes' do
            with_belongs_to_as_required_by_default do
              expect(belonging_to_parent).to belong_to(:parent)
            end
          end
        end

        context 'when belongs_to is not configured to be required by default' do
          it 'passes' do
            with_belongs_to_as_optional_by_default do
              expect(belonging_to_parent).to belong_to(:parent)
            end
          end

          context 'and a presence validation is on the attribute instead of using required: true' do
            it 'passes' do
              with_belongs_to_as_optional_by_default do
                record = belonging_to_parent do
                  validates_presence_of :parent
                end

                expect(record).to belong_to(:parent)
              end
            end
          end

          context 'and a presence validation is on the attribute with a condition' do
            context 'and the condition is true' do
              it 'passes' do
                with_belongs_to_as_optional_by_default do
                  child_model = create_child_model_belonging_to_parent do
                    attr_accessor :condition

                    validates_presence_of :parent, if: :condition
                  end

                  record = child_model.new(condition: true)

                  expect(record).to belong_to(:parent)
                end
              end
            end

            context 'and the condition is false' do
              it 'passes' do
                with_belongs_to_as_optional_by_default do
                  child_model = create_child_model_belonging_to_parent do
                    attr_accessor :condition

                    validates_presence_of :parent, if: :condition
                  end

                  record = child_model.new(condition: false)

                  expect(record).to belong_to(:parent)
                end
              end
            end
          end
        end
      end
    end

    context 'given the association is configured with required: true' do
      context 'when qualified with required(true)' do
        it 'passes' do
          expect(belonging_to_parent(required: true)).
            to belong_to(:parent).required(true)
        end
      end

      context 'when qualified with required(false)' do
        it 'fails with an appropriate message' do
          assertion = lambda do
            expect(belonging_to_parent(required: true)).
              to belong_to(:parent).required(false)
          end

          message = format_message(<<-MESSAGE, one_line: true)
            Expected Child to have a belongs_to association called parent (and
            for the record not to fail validation if :parent is unset; i.e.,
            either the association should have been defined with `required:
            false`, or there should not be a presence validation on :parent)
          MESSAGE

          expect(&assertion).to fail_with_message(message)
        end
      end

      context 'when qualified with optional(true)' do
        it 'fails with an appropriate message' do
          assertion = lambda do
            expect(belonging_to_parent(required: true)).
              to belong_to(:parent).optional(true)
          end

          message = format_message(<<-MESSAGE, one_line: true)
            Expected Child to have a belongs_to association called parent
            (and for the record not to fail validation if :parent is unset;
            i.e., either the association should have been defined with
            `optional: true`, or there should not be a presence validation on
            :parent)
          MESSAGE

          expect(&assertion).to fail_with_message(message)
        end
      end

      context 'when qualified with optional(false)' do
        it 'passes' do
          expect(belonging_to_parent(required: true)).
            to belong_to(:parent).optional(false)
        end
      end

      context 'when qualified with nothing' do
        it 'passes' do
          expect(belonging_to_parent(required: true)).to belong_to(:parent)
        end
      end
    end

    context 'given the association is configured as optional: true' do
      context 'when qualified with required(true)' do
        it 'fails with an appropriate message' do
          assertion = lambda do
            expect(belonging_to_parent(optional: true)).
              to belong_to(:parent).required(true)
          end

          message = format_message(<<-MESSAGE, one_line: true)
            Expected Child to have a belongs_to association called parent
            (and for the record to fail validation if :parent is unset; i.e.,
            either the association should have been defined with `required:
            true`, or there should be a presence validation on :parent)
          MESSAGE

          expect(&assertion).to fail_with_message(message)
        end
      end

      context 'when qualified with required(false)' do
        it 'passes' do
          expect(belonging_to_parent(optional: true)).
            to belong_to(:parent).required(false)
        end
      end

      context 'when qualified with optional(true)' do
        it 'passes' do
          expect(belonging_to_parent(optional: true)).
            to belong_to(:parent).optional(true)
        end
      end

      context 'when qualified with optional(false)' do
        it 'fails with an appropriate message' do
          assertion = lambda do
            expect(belonging_to_parent(optional: true)).
              to belong_to(:parent).optional(false)
          end

          message = format_message(<<-MESSAGE, one_line: true)
            Expected Child to have a belongs_to association called parent
            (and for the record to fail validation if :parent is unset; i.e.,
            either the association should have been defined with `optional:
            false`, or there should be a presence validation on :parent)
          MESSAGE

          expect(&assertion).to fail_with_message(message)
        end
      end

      context 'when qualified with nothing' do
        it 'fails with an appropriate message' do
          assertion = lambda do
            expect(belonging_to_parent(optional: true)).
              to belong_to(:parent)
          end

          message = format_message(<<-MESSAGE, one_line: true)
            Expected Child to have a belongs_to association called parent
            (and for the record to fail validation if :parent is unset; i.e.,
            either the association should have been defined with `required:
            true`, or there should be a presence validation on :parent)
          MESSAGE

          expect(&assertion).to fail_with_message(message)
        end
      end
    end

    context 'when the model ensures the association is set' do
      context 'and the matcher is not qualified with anything' do
        context 'and the matcher is not qualified with without_validating_presence' do
          it 'fails with an appropriate message' do
            model = create_child_model_belonging_to_parent do
              before_validation :ensure_parent_is_set

              def ensure_parent_is_set
                self.parent = Parent.create
              end
            end

            assertion = lambda do
              with_belongs_to_as_required_by_default do
                expect(model.new).to belong_to(:parent)
              end
            end

            message = format_message(<<-MESSAGE, one_line: true)
              Expected Child to have a belongs_to association called parent (and
              for the record to fail validation if :parent is unset; i.e.,
              either the association should have been defined with `required:
              true`, or there should be a presence validation on :parent)
            MESSAGE

            expect(&assertion).to fail_with_message(message)
          end
        end

        context 'and the matcher is qualified with without_validating_presence' do
          it 'passes' do
            model = create_child_model_belonging_to_parent do
              before_validation :ensure_parent_is_set

              def ensure_parent_is_set
                self.parent = Parent.create
              end
            end

            with_belongs_to_as_required_by_default do
              expect(model.new).
                to belong_to(:parent).
                without_validating_presence
            end
          end
        end
      end

      context 'and the matcher is qualified with required' do
        context 'and the matcher is not qualified with without_validating_presence' do
          it 'fails with an appropriate message' do
            model = create_child_model_belonging_to_parent do
              before_validation :ensure_parent_is_set

              def ensure_parent_is_set
                self.parent = Parent.create
              end
            end

            assertion = lambda do
              with_belongs_to_as_required_by_default do
                expect(model.new).to belong_to(:parent).required
              end
            end

            message = format_message(<<-MESSAGE, one_line: true)
              Expected Child to have a belongs_to association called parent
              (and for the record to fail validation if :parent is unset; i.e.,
              either the association should have been defined with `required:
              true`, or there should be a presence validation on :parent)
            MESSAGE

            expect(&assertion).to fail_with_message(message)
          end
        end

        context 'and the matcher is also qualified with without_validating_presence' do
          it 'passes' do
            model = create_child_model_belonging_to_parent do
              before_validation :ensure_parent_is_set

              def ensure_parent_is_set
                self.parent = Parent.create
              end
            end

            with_belongs_to_as_required_by_default do
              expect(model.new).
                to belong_to(:parent).
                required.
                without_validating_presence
            end
          end
        end
      end
    end

    unless active_record_gte_8_1?
      context 'using the deprecated matcher' do
        it 'raises a NotImplementedError' do
          expected_message = '`deprecated` association matcher is only available on Active Record >= 8.1.'

          expect do
            expect(belonging_to_parent).to belong_to(:parent).deprecated
          end.to raise_error(NotImplementedError, expected_message)
        end
      end
    end

    if active_record_gte_8_1?
      context 'an association with a :deprecated option' do
        [false, true].each do |deprecated_value|
          context "when the model has deprecated: #{deprecated_value}" do
            it 'accepts a matching deprecated option' do
              expect(belonging_to_parent(deprecated: deprecated_value)).
                to belong_to(:parent).deprecated(deprecated_value)
            end

            it 'rejects a non-matching deprecated option' do
              expect(belonging_to_parent(deprecated: deprecated_value)).
                not_to belong_to(:parent).deprecated(!deprecated_value)
            end

            it 'defaults to deprecated(true)' do
              if deprecated_value
                expect(belonging_to_parent(deprecated: deprecated_value)).
                  to belong_to(:parent).deprecated
              else
                expect(belonging_to_parent(deprecated: deprecated_value)).
                  not_to belong_to(:parent).deprecated
              end
            end

            it 'will not break matcher when deprecated option is unspecified' do
              expect(belonging_to_parent(deprecated: deprecated_value)).to belong_to(:parent)
            end
          end
        end
      end

      context 'an association without a :deprecated option' do
        it 'accepts deprecated(false)' do
          expect(belonging_to_parent).to belong_to(:parent).deprecated(false)
        end

        it 'rejects deprecated(true)' do
          expect(belonging_to_parent).not_to belong_to(:parent).deprecated(true)
        end

        it 'rejects deprecated()' do
          expect(belonging_to_parent).not_to belong_to(:parent).deprecated
        end
      end

      it 'rejects an association with a non-matching :deprecated option with the correct message' do
        define_model :parent, adopter: :boolean
        define_model :child, parent_id: :integer do
          belongs_to :parent, deprecated: false
        end

        message = 'Expected Child to have a belongs_to association called parent (parent should have deprecated set to true)'
        expect do
          expect(Child.new).to belong_to(:parent).deprecated(true)
        end.to fail_with_message(message)
      end
    end

    def belonging_to_parent(options = {}, parent_options = {}, &block)
      child_model = create_child_model_belonging_to_parent(
        options,
        parent_options,
        &block
      )
      child_model.new
    end

    def create_child_model_belonging_to_parent(
      options = {},
      parent_options = {},
      &block
    )
      define_model(:parent, parent_options)

      define_model :child, parent_id: :integer do
        belongs_to :parent, **options

        if block
          class_eval(&block)
        end
      end
    end

    def belonging_to_with_inverse(association, inverse_association)
      parent_model_name = association.to_s.singularize
      child_model_name = inverse_association.to_s.singularize
      parent_foreign_key = "#{parent_model_name}_id"

      define_model parent_model_name do
        has_many inverse_association
      end

      child_model = define_model(
        child_model_name,
        parent_foreign_key => :integer,
      ) do
        belongs_to association, inverse_of: inverse_association
      end

      child_model.new
    end

    def belonging_to_non_existent_class(model_name, assoc_name, options = {})
      define_model model_name, "#{assoc_name}_id" => :integer do
        belongs_to assoc_name, **options
      end.new
    end
  end

  context 'have_many' do
    unless active_record_gte_8_1?
      context 'using the deprecated matcher' do
        it 'raises a NotImplementedError' do
          expected_message = '`deprecated` association matcher is only available on Active Record >= 8.1.'

          expect do
            expect(having_many_children).to have_many(:children).deprecated
          end.to raise_error(NotImplementedError, expected_message)
        end
      end
    end

    it 'accepts a valid association without any options' do
      expect(having_many_children).to have_many(:children)
    end

    it 'does not reject a non-:through association where there is no belongs_to in the inverse model' do
      define_model :Child, parent_id: :integer
      parent_class = define_model :Parent do
        has_many :children
      end

      expect { have_many(:children) }.to match_against(parent_class.new)
    end

    it 'accepts a valid association with a :through option' do
      define_model :child
      define_model :conception, child_id: :integer, parent_id: :integer do
        belongs_to :child
      end
      define_model :parent do
        has_many :conceptions
        has_many :children, through: :conceptions
      end
      expect(Parent.new).to have_many(:children)
    end

    it 'rejects a :through association where there is no belongs_to in the inverse model' do
      define_model :Child
      define_model :Conception, child_id: :integer, parent_id: :integer
      parent_class = define_model :Parent do
        has_many :conceptions
        has_many :children, through: :conceptions
      end

      expect { have_many(:children) }.not_to match_against(parent_class.new).and_fail_with(<<-MESSAGE)
Expected Parent to have a has_many association called children through conceptions (Could not find the source association(s) "child" or :children in model Conception. Try 'has_many :children, :through => :conceptions, :source => <name>'. Is it one of ?)
      MESSAGE
    end

    it 'accepts a valid association with an :as option' do
      define_model :child, guardian_type: :string, guardian_id: :integer
      define_model :parent do
        has_many :children, as: :guardian
      end

      expect(Parent.new).to have_many(:children)
    end

    it 'rejects an association that has a nonexistent foreign key' do
      define_model :child
      define_model :parent do
        has_many :children
      end

      expect(Parent.new).not_to have_many(:children)
    end

    it 'accepts an association using an existing custom primary key' do
      define_model :child, parent_id: :integer
      define_model :parent, custom_primary_key: :integer do
        has_many :children, primary_key: :custom_primary_key
      end
      expect(Parent.new).to have_many(:children).with_primary_key(:custom_primary_key)
    end

    it 'rejects an association with a bad :primary_key option' do
      matcher = have_many(:children).with_primary_key(:custom_primary_key)

      expect(having_many_children).not_to matcher

      expect(matcher.failure_message).to match(/Parent does not have a custom_primary_key primary key/)
    end

    if rails_version >= 7.1 && !rails_gt_8_0?
      it 'accepts an association using an valid :query_constraints option' do
        define_model :parent, first_name: :string, last_name: :string do
          self.primary_key = [:first_name, :last_name]

          has_many :children, query_constraints: [:parent_first_name, :parent_last_name]
        end

        define_model :child, parent_first_name: :integer, parent_last_name: :string do
          belongs_to :parent, query_constraints: [:parent_first_name, :parent_last_name]
        end

        expect(Parent.new).to have_many(:children).with_query_constraints([:parent_first_name, :parent_last_name])
      end

      it 'rejects an association with a bad :query_constraints option' do
        matcher = have_many(:children).with_query_constraints([:parent_first_name, :parent_last_name])

        expect(having_many_children).not_to matcher

        expect(matcher.failure_message).to match(/Parent should have :query_constraints options set to \[:parent_first_name, :parent_last_name\]/)
      end
    end

    it 'rejects an association with a bad :as option' do
      define_model(
        :child,
        caretaker_type: :string,
        caretaker_id: :integer,
      )
      define_model :parent do
        has_many :children, as: :guardian
      end

      expect(Parent.new).not_to have_many(:children)
    end

    it 'rejects an association that has a bad :through option' do
      matcher = have_many(:children).through(:conceptions)

      expect(matcher.matches?(having_many_children)).to eq false

      expect(matcher.failure_message).to match(/does not have any relationship to conceptions/)
    end

    it 'rejects an association that has the wrong :through option' do
      define_model :child

      define_model :conception, child_id: :integer, parent_id: :integer do
        belongs_to :child
      end

      define_model :parent do
        has_many :conceptions
        has_many :relationships
        has_many :children, through: :conceptions
      end

      matcher = have_many(:children).through(:relationships)
      expect(matcher.matches?(Parent.new)).to eq false
      expect(matcher.failure_message).to match(/through relationships, but got it through conceptions/)
    end

    it 'produces a failure message without exception when association is missing :through option' do
      define_model :child
      define_model :parent
      matcher = have_many(:children).through(:relationships).source(:child)
      failure_message = 'Expected Parent to have a has_many association called children (no association called children)'

      matcher.matches?(Parent.new)
      expect(matcher.failure_message).to eq failure_message
    end

    it 'accepts an association with a valid :dependent option' do
      expect(having_many_children(dependent: :destroy)).
        to have_many(:children).dependent(:destroy)
    end

    it 'rejects an association with a bad :dependent option' do
      matcher = have_many(:children).dependent(:destroy)

      expect(having_many_children).not_to matcher

      expect(matcher.failure_message).to match(/children should have destroy dependency/)
    end

    it 'accepts an association with a valid :source option' do
      define_model(:author) do
        has_many :books
        has_many :paperbacks, through: :books, source: :format, source_type: 'Paperback'
      end
      define_model(:book, format_id: :integer) do
        belongs_to :format, polymorphic: true
      end
      define_model(:paperback)

      expect(Author.new).to have_many(:paperbacks).source(:format)
    end

    it 'rejects an association with a bad :source option' do
      matcher = have_many(:children).source(:user)

      expect(having_many_children).not_to matcher

      expect(matcher.failure_message).to match(/children should have user as source option/)
    end

    it 'accepts an association with a valid :order option' do
      expect(having_many_children(order: :id)).
        to have_many(:children).order(:id)
    end

    it 'rejects an association with a bad :order option' do
      matcher = have_many(:children).order(:id)

      expect(having_many_children).not_to matcher

      expect(matcher.failure_message).to match(/children should be ordered by id/)
    end

    context 'if the association has a scope block' do
      context 'and the block does not take an argument' do
        context 'and the matcher is given conditions that match the conditions used in the scope' do
          it 'matches' do
            define_model :Child, parent_id: :integer, adopted: :boolean
            define_model(:Parent) do
              has_many :children, -> { where(adopted: true) }
            end

            expect(Parent.new).
              to have_many(:children).
              conditions(adopted: true)
          end
        end

        context 'and the matcher is given conditions that do not match the conditions used in the scope' do
          it 'rejects an association with a bad :conditions option' do
            define_model :Child, parent_id: :integer, adopted: :boolean
            define_model :Parent do
              has_many :children
            end

            expect(Parent.new).
              not_to have_many(:children).
              conditions(adopted: true)
          end
        end
      end

      context 'and the block takes an argument' do
        context 'and the matcher is given conditions that match the scope' do
          it 'matches' do
            define_model :Wheel, bike_id: :integer, tire_id: :integer
            define_model :Tire
            define_model :Bike, default_tire_id: :integer do
              has_many :wheels, -> (bike) do
                where(tire_id: bike.default_tire_id)
              end
              belongs_to :default_tire, class_name: 'Tire'
            end

            expect(Bike.new(default_tire_id: 42)).
              to have_many(:wheels).conditions(tire_id: 42)
          end
        end

        context 'and the matcher is given conditions that do not match the scope' do
          it 'matches' do
            define_model :Wheel, bike_id: :integer, tire_id: :integer
            define_model :Tire
            define_model :Bike, default_tire_id: :integer do
              has_many :wheels, -> (bike) do
                where(tire_id: bike.default_tire_id)
              end
              belongs_to :default_tire, class_name: 'Tire'
            end

            expect(Bike.new(default_tire_id: 42)).
              not_to have_many(:wheels).conditions(tire_id: 10)
          end
        end
      end
    end

    it 'accepts an association without a :class_name option' do
      expect(having_many_children).to have_many(:children).class_name('Child')
    end

    it 'accepts an association with a valid :class_name option' do
      define_model :node, parent_id: :integer
      define_model :parent do
        has_many :children, class_name: 'Node'
      end

      expect(Parent.new).to have_many(:children).class_name('Node')
    end

    it 'rejects an association with a bad :class_name option' do
      expect(having_many_children).not_to have_many(:children).class_name('Node')
    end

    it 'rejects an association with non-existent implicit class name' do
      expect(having_many_non_existent_class(:parent, :children)).not_to have_many(:children)
    end

    it 'rejects an association with non-existent explicit class name' do
      expect(having_many_non_existent_class(:parent, :children, class_name: 'Child')).not_to have_many(:children)
    end

    it 'adds error message when rejecting an association with non-existent class' do
      message = 'Expected Parent to have a has_many association called children (Child2 does not exist)'
      expect {
        expect(having_many_non_existent_class(:parent, :children, class_name: 'Child2')).to have_many(:children)
      }.to fail_with_message(message)
    end

    it 'accepts an association with a namespaced class name' do
      define_module 'Models'
      define_model 'Models::Friend', user_id: :integer
      friend_model = define_model 'Models::User' do
        has_many :friends, class_name: 'Friend'
      end

      expect(friend_model.new).
        to have_many(:friends).
        class_name('Friend')
    end

    it 'resolves class_name within the context of the namespace before the global namespace' do
      define_module 'Models'
      define_model 'Friend'
      define_model 'Models::Friend', user_id: :integer
      friend_model = define_model 'Models::User' do
        has_many :friends, class_name: 'Friend'
      end

      expect(friend_model.new).
        to have_many(:friends).
        class_name('Friend')
    end

    it 'accepts an association with a matching :autosave option' do
      define_model :child, parent_id: :integer
      define_model :parent do
        has_many :children, autosave: true
      end
      expect(Parent.new).to have_many(:children).autosave(true)
    end

    it 'rejects an association with a non-matching :autosave option with the correct message' do
      define_model :child, parent_id: :integer
      define_model :parent do
        has_many :children, autosave: false
      end

      message = 'Expected Parent to have a has_many association called children (children should have autosave set to true)'
      expect {
        expect(Parent.new).to have_many(:children).autosave(true)
      }.to fail_with_message(message)
    end

    context 'index_errors' do
      it 'accepts an association with a matching :index_errors option' do
        define_model :child, parent_id: :integer
        define_model :parent do
          has_many :children, index_errors: true
        end
        expect(Parent.new).to have_many(:children).index_errors(true)
      end

      it 'rejects an association with a non-matching :index_errors option and returns the correct message' do
        define_model :child, parent_id: :integer
        define_model :parent do
          has_many :children, autosave: false
        end

        message =
          'Expected Parent to have a has_many association called children '\
          '(children should have index_errors set to true)'

        expect {
          expect(Parent.new).to have_many(:children).index_errors(true)
        }.to fail_with_message(message)
      end
    end

    context 'validate' do
      it 'accepts validate(false) when the :validate option is false' do
        expect(having_many_children(validate: false)).to have_many(:children).validate(false)
      end

      it 'accepts validate(true) when the :validate option is true' do
        expect(having_many_children(validate: true)).to have_many(:children).validate(true)
      end

      it 'rejects validate(false) when the :validate option is true' do
        expect(having_many_children(validate: true)).not_to have_many(:children).validate(false)
      end

      it 'rejects validate(true) when the :validate option is false' do
        expect(having_many_children(validate: false)).not_to have_many(:children).validate(true)
      end

      it 'assumes validate() means validate(true)' do
        expect(having_many_children(validate: true)).to have_many(:children).validate
      end

      it 'rejects validate() when :validate option is false' do
        expect(having_many_children(validate: false)).not_to have_many(:children).validate
      end

      it 'rejects validate(false) when no :validate option was specified' do
        expect(having_many_children).not_to have_many(:children).validate(false)
      end

      it 'accepts validate(true) when no :validate option was specified' do
        expect(having_many_children).to have_many(:children).validate(true)
      end

      it 'accepts validate() when no :validate option was specified' do
        expect(having_many_children).to have_many(:children).validate
      end
    end

    it 'accepts an association with a nonstandard reverse foreign key, using :inverse_of' do
      define_model :child, ancestor_id: :integer do
        belongs_to :ancestor, inverse_of: :children, class_name: :Parent
      end

      define_model :parent do
        has_many :children, inverse_of: :ancestor
      end

      expect(Parent.new).to have_many(:children)
    end

    it 'accepts an association with a nonstandard reverse foreign type, using :inverse_of' do
      define_model :visitor, location_id: :integer, facility_type: :string do
        belongs_to :location, foreign_type: :facility_type, inverse_of: :visitors, polymorphic: true
      end

      define_model :hotel do
        has_many :visitors, inverse_of: :location, foreign_type: :facility_type, as: :location
      end

      expect(Hotel.new).to have_many(:visitors).with_foreign_type(:facility_type)
    end

    it 'rejects an association with a nonstandard reverse foreign key, if :inverse_of is not correct' do
      define_model :child, mother_id: :integer do
        belongs_to :mother, inverse_of: :children, class_name: :Parent
      end

      define_model :parent do
        has_many :children, inverse_of: :ancestor
      end

      expect(Parent.new).not_to have_many(:children)
    end

    it 'accepts an association with a nonstandard foreign key, with reverse association turned off' do
      define_model :child, ancestor_id: :integer
      define_model :parent do
        has_many :children, foreign_key: :ancestor_id, inverse_of: false
      end

      expect(Parent.new).to have_many(:children)
    end

    it 'accepts an association with a nonstandard type, with reverse association turned off' do
      define_model :visitor, location_id: :integer, facility_type: :string

      define_model :hotel do
        has_many :visitors, foreign_type: :facility_type, inverse_of: false, as: :location
      end

      expect(Hotel.new).to have_many(:visitors).with_foreign_type(:facility_type)
    end

    if rails_version >= 8.1
      context 'deprecated' do
        it 'accepts deprecated(false) when the :deprecated option is false' do
          expect(having_many_children(deprecated: false)).to have_many(:children).deprecated(false)
        end

        it 'accepts deprecated(true) when the :deprecated option is true' do
          expect(having_many_children(deprecated: true)).to have_many(:children).deprecated(true)
        end

        it 'rejects deprecated(false) when the :deprecated option is true' do
          expect(having_many_children(deprecated: true)).not_to have_many(:children).deprecated(false)
        end

        it 'rejects deprecated(true) when the :deprecated option is false' do
          expect(having_many_children(deprecated: false)).not_to have_many(:children).deprecated(true)
        end

        it 'assumes deprecated() means deprecated(true)' do
          expect(having_many_children(deprecated: true)).to have_many(:children).deprecated
        end

        it 'rejects deprecated() when :deprecated option is false' do
          expect(having_many_children(deprecated: false)).not_to have_many(:children).deprecated
        end

        it 'rejects deprecated(true) when no :deprecated option was specified' do
          expect(having_many_children).not_to have_many(:children).deprecated(true)
        end

        it 'rejects deprecated(false) when no :deprecated option was specified' do
          expect(having_many_children).to have_many(:children).deprecated(false)
        end

        it 'rejects deprecated() when no :deprecated option was specified' do
          expect(having_many_children).not_to have_many(:children).deprecated
        end

        it 'accepts an association :through with a :deprecated option' do
          define_model(:author) do
            has_many :books
            has_many :paperbacks, through: :books, source: :format, source_type: 'Paperback', deprecated: true
          end
          define_model(:book, format_id: :integer) do
            belongs_to :format, polymorphic: true
          end
          define_model(:paperback)

          expect(Author.new).to have_many(:paperbacks).source(:format).deprecated
          expect(Author.new).not_to have_many(:paperbacks).source(:format).deprecated(false)
        end

        it 'accepts an association :through without a :deprecated option' do
          define_model(:author) do
            has_many :books
            has_many :paperbacks, through: :books, source: :format, source_type: 'Paperback'
          end
          define_model(:book, format_id: :integer) do
            belongs_to :format, polymorphic: true
          end
          define_model(:paperback)

          expect(Author.new).to have_many(:paperbacks).source(:format).deprecated(false)
          expect(Author.new).not_to have_many(:paperbacks).source(:format).deprecated(true)
        end

        it 'rejects an association with a non-matching :deprecated option with the correct message' do
          define_model :child, parent_id: :integer
          define_model :parent do
            has_many :children, deprecated: false
          end

          message = 'Expected Parent to have a has_many association called children (children should have deprecated set to true)'
          expect do
            expect(Parent.new).to have_many(:children).deprecated(true)
          end.to fail_with_message(message)
        end
      end
    end

    describe 'strict_loading' do
      context 'when the application is configured with strict_loading disabled by default' do
        it 'accepts an association with a matching :strict_loading option' do
          with_strict_loading_by_default_disabled do
            expect(having_many_children).
              to have_many(:children).strict_loading(false)
          end
        end

        it 'rejects an association with a non-matching :strict_loading option without explicit value with the correct message' do
          with_strict_loading_by_default_disabled do
            message = [
              'Expected Parent to have a has_many association called children ',
              '(children should have strict_loading set to true)',
            ].join

            expect {
              expect(having_many_children).
                to have_many(:children).strict_loading
            }.to fail_with_message(message)
          end
        end

        it 'rejects an association with a non-matching :strict_loading option with the correct message' do
          with_strict_loading_by_default_disabled do
            message = [
              'Expected Parent to have a has_many association called children ',
              '(children should have strict_loading set to true)',
            ].join

            expect {
              expect(having_many_children).
                to have_many(:children).strict_loading(true)
            }.to fail_with_message(message)
          end
        end

        context 'when the association is configured with a strict_loading constraint' do
          context 'when qualified with strict_loading(true)' do
            it 'accepts an association with a matching :strict_loading option' do
              with_strict_loading_by_default_disabled do
                expect(having_many_children(strict_loading: true)).
                  to have_many(:children).strict_loading(true)
              end
            end

            it 'accepts an association with a matching :strict_loading option without explicit value' do
              with_strict_loading_by_default_disabled do
                expect(having_many_children(strict_loading: true)).
                  to have_many(:children).strict_loading
              end
            end

            it 'rejects an association with a non-matching :strict_loading option with the correct message' do
              with_strict_loading_by_default_disabled do
                message = [
                  'Expected Parent to have a has_many association called children ',
                  '(children should have strict_loading set to false)',
                ].join

                expect {
                  expect(having_many_children(strict_loading: true)).
                    to have_many(:children).strict_loading(false)
                }.to fail_with_message(message)
              end
            end
          end

          context 'when qualified with strict_loading(false)' do
            it 'accepts an association with a matching :strict_loading option' do
              with_strict_loading_by_default_disabled do
                expect(having_many_children(strict_loading: false)).
                  to have_many(:children).strict_loading(false)
              end
            end

            it 'rejects an association with a non-matching :strict_loading option without explicit value with the correct message' do
              with_strict_loading_by_default_disabled do
                message = [
                  'Expected Parent to have a has_many association called children ',
                  '(children should have strict_loading set to true)',
                ].join

                expect {
                  expect(having_many_children(strict_loading: false)).
                    to have_many(:children).strict_loading
                }.to fail_with_message(message)
              end
            end

            it 'rejects an association with a non-matching :strict_loading option with the correct message' do
              with_strict_loading_by_default_disabled do
                message = [
                  'Expected Parent to have a has_many association called children ',
                  '(children should have strict_loading set to true)',
                ].join

                expect {
                  expect(having_many_children(strict_loading: false)).
                    to have_many(:children).strict_loading(true)
                }.to fail_with_message(message)
              end
            end
          end
        end

        context 'when strict_loading is defined on the model level' do
          context 'when it is set to true' do
            it 'accepts an association with a matching :strict_loading option' do
              with_strict_loading_by_default_disabled do
                define_model :child, parent_id: :integer
                parent = define_model(:parent) do |model|
                  model.strict_loading_by_default = true
                  model.has_many :children
                end.new

                expect(parent).to have_many(:children).strict_loading(true)
              end
            end

            it 'accepts an association with a matching :strict_loading option without explicit value' do
              with_strict_loading_by_default_disabled do
                define_model :child, parent_id: :integer
                parent = define_model(:parent) do |model|
                  model.strict_loading_by_default = true
                  model.has_many :children
                end.new

                expect(parent).to have_many(:children).strict_loading
              end
            end

            it 'rejects an association with a non-matching :strict_loading option with the correct message' do
              with_strict_loading_by_default_disabled do
                message = [
                  'Expected Parent to have a has_many association called children ',
                  '(children should have strict_loading set to false)',
                ].join

                define_model :child, parent_id: :integer
                parent = define_model(:parent) do |model|
                  model.strict_loading_by_default = true
                  model.has_many :children
                end.new

                expect {
                  expect(parent).to have_many(:children).strict_loading(false)
                }.to fail_with_message(message)
              end
            end
          end

          context 'when it is set to false' do
            it 'accepts an association with a matching :strict_loading option' do
              with_strict_loading_by_default_disabled do
                define_model :child, parent_id: :integer
                parent = define_model(:parent) do |model|
                  model.strict_loading_by_default = false
                  model.has_many :children
                end.new

                expect(parent).to have_many(:children).strict_loading(false)
              end
            end

            it 'rejects an association with a non-matching :strict_loading option without explicit value with the correct message' do
              with_strict_loading_by_default_disabled do
                message = [
                  'Expected Parent to have a has_many association called children ',
                  '(children should have strict_loading set to true)',
                ].join

                define_model :child, parent_id: :integer
                parent = define_model(:parent) do |model|
                  model.strict_loading_by_default = false
                  model.has_many :children
                end.new

                expect {
                  expect(parent).to have_many(:children).strict_loading
                }.to fail_with_message(message)
              end
            end

            it 'rejects an association with a non-matching :strict_loading option with the correct message' do
              with_strict_loading_by_default_disabled do
                message = [
                  'Expected Parent to have a has_many association called children ',
                  '(children should have strict_loading set to true)',
                ].join

                define_model :child, parent_id: :integer
                parent = define_model(:parent) do |model|
                  model.strict_loading_by_default = false
                  model.has_many :children
                end.new

                expect {
                  expect(parent).to have_many(:children).strict_loading(true)
                }.to fail_with_message(message)
              end
            end
          end
        end
      end

      context 'when the application is configured with strict_loading enabled by default' do
        it 'accepts an association with a matching :strict_loading option' do
          with_strict_loading_by_default_enabled do
            expect(having_many_children).
              to have_many(:children).strict_loading(true)
          end
        end

        it 'accepts an association with a matching :strict_loading option without explicit value' do
          with_strict_loading_by_default_enabled do
            expect(having_many_children).
              to have_many(:children).strict_loading
          end
        end

        it 'rejects an association with a non-matching :strict_loading option with the correct message' do
          with_strict_loading_by_default_enabled do
            message = [
              'Expected Parent to have a has_many association called children ',
              '(children should have strict_loading set to false)',
            ].join

            expect {
              expect(having_many_children).
                to have_many(:children).strict_loading(false)
            }.to fail_with_message(message)
          end
        end

        context 'when the association is configured with a strict_loading constraint' do
          context 'when qualified with strict_loading(true)' do
            it 'accepts an association with a matching :strict_loading option' do
              with_strict_loading_by_default_enabled do
                expect(having_many_children(strict_loading: true)).
                  to have_many(:children).strict_loading(true)
              end
            end

            it 'accepts an association with a matching :strict_loading option without explicit value' do
              with_strict_loading_by_default_enabled do
                expect(having_many_children(strict_loading: true)).
                  to have_many(:children).strict_loading
              end
            end

            it 'rejects an association with a non-matching :strict_loading option with the correct message' do
              with_strict_loading_by_default_enabled do
                message = [
                  'Expected Parent to have a has_many association called children ',
                  '(children should have strict_loading set to false)',
                ].join

                expect {
                  expect(having_many_children(strict_loading: true)).
                    to have_many(:children).strict_loading(false)
                }.to fail_with_message(message)
              end
            end
          end

          context 'when qualified with strict_loading(false)' do
            it 'accepts an association with a matching :strict_loading option' do
              with_strict_loading_by_default_enabled do
                expect(having_many_children(strict_loading: false)).
                  to have_many(:children).strict_loading(false)
              end
            end

            it 'rejects an association with a non-matching :strict_loading option without explicit value with the correct message' do
              with_strict_loading_by_default_enabled do
                message = [
                  'Expected Parent to have a has_many association called children ',
                  '(children should have strict_loading set to true)',
                ].join

                expect {
                  expect(having_many_children(strict_loading: false)).
                    to have_many(:children).strict_loading
                }.to fail_with_message(message)
              end
            end

            it 'rejects an association with a non-matching :strict_loading option with the correct message' do
              with_strict_loading_by_default_enabled do
                message = [
                  'Expected Parent to have a has_many association called children ',
                  '(children should have strict_loading set to true)',
                ].join

                expect {
                  expect(having_many_children(strict_loading: false)).
                    to have_many(:children).strict_loading(true)
                }.to fail_with_message(message)
              end
            end
          end
        end

        context 'when strict_loading is defined on the model level' do
          context 'when it is set to true' do
            it 'accepts an association with a matching :strict_loading option' do
              with_strict_loading_by_default_enabled do
                define_model :child, parent_id: :integer
                parent = define_model(:parent) do |model|
                  model.strict_loading_by_default = true
                  model.has_many :children
                end.new

                expect(parent).to have_many(:children).strict_loading(true)
              end
            end

            it 'accepts an association with a matching :strict_loading option without explicit value' do
              with_strict_loading_by_default_enabled do
                define_model :child, parent_id: :integer
                parent = define_model(:parent) do |model|
                  model.strict_loading_by_default = true
                  model.has_many :children
                end.new

                expect(parent).to have_many(:children).strict_loading
              end
            end

            it 'rejects an association with a non-matching :strict_loading option with the correct message' do
              with_strict_loading_by_default_enabled do
                message = [
                  'Expected Parent to have a has_many association called children ',
                  '(children should have strict_loading set to false)',
                ].join

                define_model :child, parent_id: :integer
                parent = define_model(:parent) do |model|
                  model.strict_loading_by_default = true
                  model.has_many :children
                end.new

                expect {
                  expect(parent).to have_many(:children).strict_loading(false)
                }.to fail_with_message(message)
              end
            end
          end

          context 'when it is set to false' do
            it 'accepts an association with a matching :strict_loading option' do
              with_strict_loading_by_default_enabled do
                define_model :child, parent_id: :integer
                parent = define_model(:parent) do |model|
                  model.strict_loading_by_default = false
                  model.has_many :children
                end.new

                expect(parent).to have_many(:children).strict_loading(false)
              end
            end

            it 'rejects an association with a non-matching :strict_loading option without explicit value with the correct message' do
              with_strict_loading_by_default_enabled do
                message = [
                  'Expected Parent to have a has_many association called children ',
                  '(children should have strict_loading set to true)',
                ].join

                define_model :child, parent_id: :integer
                parent = define_model(:parent) do |model|
                  model.strict_loading_by_default = false
                  model.has_many :children
                end.new

                expect {
                  expect(parent).to have_many(:children).strict_loading
                }.to fail_with_message(message)
              end
            end

            it 'rejects an association with a non-matching :strict_loading option with the correct message' do
              with_strict_loading_by_default_enabled do
                message = [
                  'Expected Parent to have a has_many association called children ',
                  '(children should have strict_loading set to true)',
                ].join

                define_model :child, parent_id: :integer
                parent = define_model(:parent) do |model|
                  model.strict_loading_by_default = false
                  model.has_many :children
                end.new

                expect {
                  expect(parent).to have_many(:children).strict_loading(true)
                }.to fail_with_message(message)
              end
            end
          end
        end
      end
    end

    def having_many_children(options = {})
      define_model :child, parent_id: :integer
      define_model(:parent).tap do |model|
        if options.key?(:order)
          order = options.delete(:order)
          define_association_with_order(model, :has_many, :children, order, options)
        else
          model.has_many :children, **options
        end
      end.new
    end

    def having_many_non_existent_class(model_name, assoc_name, options = {})
      define_model model_name do
        has_many assoc_name, **options
      end.new
    end
  end

  context 'have_one' do
    it 'accepts a valid association without any options' do
      expect(having_one_detail).to have_one(:detail)
    end

    it 'accepts a valid association with an :as option' do
      define_model :detail, detailable_id: :integer,
        detailable_type: :string
      define_model :person do
        has_one :detail, as: :detailable
      end

      expect(Person.new).to have_one(:detail)
    end

    it 'rejects an association that has a nonexistent foreign key' do
      define_model :detail
      define_model :person do
        has_one :detail
      end

      expect(Person.new).not_to have_one(:detail)
    end

    it 'accepts an association with an existing custom foreign key' do
      define_model :detail, detailed_person_id: :integer
      define_model :person do
        has_one :detail, foreign_key: :detailed_person_id
      end
      expect(Person.new).to have_one(:detail).with_foreign_key(:detailed_person_id)
    end

    it 'accepts an association with an existing custom foreign type' do
      define_model :profile, user_id: :integer, related_user_type: :string

      define_model :admin do
        has_one :profile, foreign_type: :related_user_type, as: :user
      end

      define_model :moderator do
        has_one :profile, foreign_type: :related_user_type, as: :user
      end

      expect(Admin.new).to have_one(:profile).with_foreign_type(:related_user_type)
      expect(Moderator.new).to have_one(:profile).with_foreign_type(:related_user_type)
    end

    it 'accepts an association using an existing custom primary key' do
      define_model :detail, person_id: :integer
      define_model :person, custom_primary_key: :integer do
        has_one :detail, primary_key: :custom_primary_key
      end
      expect(Person.new).to have_one(:detail).with_primary_key(:custom_primary_key)
    end

    it 'rejects an association with a bad :primary_key option' do
      matcher = have_one(:detail).with_primary_key(:custom_primary_key)

      expect(having_one_detail).not_to matcher

      expect(matcher.failure_message).to match(/Person does not have a custom_primary_key primary key/)
    end

    if rails_version >= 7.1 && !rails_gt_8_0?
      it 'accepts an association using an valid :query_constraints option' do
        define_model :detail, person_first_name: :string, person_last_name: :string
        define_model :person do
          has_one :detail, query_constraints: [:person_first_name, :person_last_name]
        end

        expect(Person.new).to have_one(:detail).with_query_constraints([:person_first_name, :person_last_name])
      end

      it 'rejects an association with a bad :query_constraints option' do
        matcher = have_one(:detail).with_query_constraints([:person_first_name, :person_last_name])

        expect(having_one_detail).not_to matcher

        expect(matcher.failure_message).to match(/Person should have :query_constraints options set to \[:person_first_name, :person_last_name\]/)
      end
    end

    it 'rejects an association with a bad :as option' do
      define_model :detail, detailable_id: :integer,
        detailable_type: :string
      define_model :person do
        has_one :detail, as: :describable
      end

      expect(Person.new).not_to have_one(:detail)
    end

    it 'accepts an association with a valid :dependent option' do
      dependent_options.each do |option|
        expect(having_one_detail(dependent: option)).
          to have_one(:detail).dependent(option)
      end
    end

    it 'accepts any dependent option if true' do
      dependent_options.each do |option|
        expect(having_one_detail(dependent: option)).
          to have_one(:detail).dependent(true)
      end
    end

    it 'rejects any dependent options if false' do
      dependent_options.each do |option|
        expect(having_one_detail(dependent: option)).
          to_not have_one(:detail).dependent(false)
      end
    end

    it 'accepts a nil dependent option if false' do
      expect(having_one_detail).to have_one(:detail).dependent(false)
    end

    it 'rejects an association with a bad :dependent option' do
      matcher = have_one(:detail).dependent(:destroy)

      expect(having_one_detail).not_to matcher

      expect(matcher.failure_message).to match(/detail should have destroy dependency/)
    end

    it 'accepts an association with a valid :order option' do
      expect(having_one_detail(order: :id)).to have_one(:detail).order(:id)
    end

    it 'rejects an association with a bad :order option' do
      matcher = have_one(:detail).order(:id)

      expect(having_one_detail).not_to matcher

      expect(matcher.failure_message).to match(/detail should be ordered by id/)
    end

    it 'accepts an association with a valid :conditions option' do
      define_model :detail, person_id: :integer, disabled: :boolean
      define_model(:person).tap do |model|
        define_association_with_conditions(model, :has_one, :detail, disabled: true)
      end

      expect(Person.new).to have_one(:detail).conditions(disabled: true)
    end

    it 'rejects an association with a bad :conditions option' do
      define_model :detail, person_id: :integer, disabled: :boolean
      define_model :person do
        has_one :detail
      end

      expect(Person.new).not_to have_one(:detail).conditions(disabled: true)
    end

    it 'accepts an association without a :class_name option' do
      expect(having_one_detail).to have_one(:detail).class_name('Detail')
    end

    it 'accepts an association with a valid :class_name option' do
      define_model :person_detail, person_id: :integer
      define_model :person do
        has_one :detail, class_name: 'PersonDetail'
      end

      expect(Person.new).to have_one(:detail).class_name('PersonDetail')
    end

    it 'rejects an association with a bad :class_name option' do
      expect(having_one_detail).not_to have_one(:detail).class_name('NotSet')
    end

    it 'rejects an association with non-existent implicit class name' do
      expect(having_one_non_existent(:pserson, :detail)).not_to have_one(:detail)
    end

    it 'rejects an association with non-existent explicit class name' do
      expect(having_one_non_existent(:person, :detail, class_name: 'Detail')).not_to have_one(:detail)
    end

    it 'rejects an association with a valid :class_name and a bad :foreign_key option' do
      define_model :person_detail
      define_model :person do
        has_one :detail, class_name: 'PersonDetail', foreign_key: :person_detail_id
      end

      expected_message = 'Expected Person to have a has_one association called detail ' \
        '(PersonDetail does not have a custom_primary_id foreign key.)'

      expect { have_one(:detail).class_name('PersonDetail').with_foreign_key(:custom_primary_id) }.
        not_to match_against(Person.new).
        and_fail_with(expected_message)
    end

    it 'adds error message when rejecting an association with non-existent class' do
      message = 'Expected Person to have a has_one association called detail (Detail2 does not exist)'
      expect {
        expect(having_one_non_existent(:person, :detail, class_name: 'Detail2')).to have_one(:detail)
      }.to fail_with_message(message)
    end

    it 'accepts an association with a namespaced class name' do
      define_module 'Models'
      define_model 'Models::Account', user_id: :integer
      user_model = define_model 'Models::User' do
        has_one :account, class_name: 'Account'
      end

      expect(user_model.new).
        to have_one(:account).
        class_name('Account')
    end

    it 'resolves class_name within the context of the namespace before the global namespace' do
      define_module 'Models'
      define_model 'Account'
      define_model 'Models::Account', user_id: :integer
      user_model = define_model 'Models::User' do
        has_one :account, class_name: 'Account'
      end

      expect(user_model.new).
        to have_one(:account).
        class_name('Account')
    end

    it 'accepts an association with a matching :autosave option' do
      define_model :detail, person_id: :integer, disabled: :boolean
      define_model :person do
        has_one :detail, autosave: true
      end
      expect(Person.new).to have_one(:detail).autosave(true)
    end

    it 'rejects an association with a non-matching :autosave option with the correct message' do
      define_model :detail, person_id: :integer, disabled: :boolean
      define_model :person do
        has_one :detail, autosave: false
      end

      message = 'Expected Person to have a has_one association called detail (detail should have autosave set to true)'
      expect {
        expect(Person.new).to have_one(:detail).autosave(true)
      }.to fail_with_message(message)
    end

    if rails_version >= 8.1
      it 'accepts an association with a matching :deprecated option' do
        define_model :detail, person_id: :integer, disabled: :boolean
        define_model :person do
          has_one :detail, deprecated: true
        end

        expect(Person.new).to have_one(:detail).deprecated(true)
      end

      it 'rejects an association with a non-matching :deprecated option' do
        define_model :detail, person_id: :integer, disabled: :boolean
        define_model :person do
          has_one :detail, deprecated: false
        end

        expect(Person.new).to have_one(:detail).deprecated(false)
      end

      it 'rejects an association with a non-matching :deprecated option when no option is passed' do
        define_model :detail, person_id: :integer, disabled: :boolean
        define_model :person do
          has_one :detail
        end

        expect(Person.new).to have_one(:detail).deprecated(false)
      end

      it 'accepts an association :through with a :deprecated option' do
        define_model :detail

        define_model :account do
          has_one :detail
        end

        define_model :person do
          has_one :account
          has_one :detail, through: :account, deprecated: true
        end

        expect(Person.new).to have_one(:detail).through(:account).deprecated
        expect(Person.new).not_to have_one(:detail).through(:account).deprecated(false)
      end

      it 'accepts an association :through without a :deprecated option' do
        define_model :detail

        define_model :account do
          has_one :detail
        end

        define_model :person do
          has_one :account
          has_one :detail, through: :account
        end

        expect(Person.new).to have_one(:detail).through(:account).deprecated(false)
        expect(Person.new).not_to have_one(:detail).through(:account).deprecated(true)
      end

      it 'rejects an association with a non-matching :deprecated option with the correct message' do
        define_model :detail, person_id: :integer, disabled: :boolean
        define_model :person do
          has_one :detail, deprecated: false
        end

        message = 'Expected Person to have a has_one association called detail (detail should have deprecated set to true)'
        expect do
          expect(Person.new).to have_one(:detail).deprecated(true)
        end.to fail_with_message(message)
      end
    end

    it 'accepts an association with a through' do
      define_model :detail

      define_model :account do
        has_one :detail
      end

      define_model :person do
        has_one :account
        has_one :detail, through: :account
      end

      expect(Person.new).to have_one(:detail).through(:account)
    end

    it 'rejects an association with a bad through' do
      expect(having_one_detail).not_to have_one(:detail).through(:account)
    end

    context 'validate' do
      it 'accepts when the :validate option matches' do
        expect(having_one_detail(validate: false)).
          to have_one(:detail).validate(false)
      end

      it 'rejects when the :validate option does not match' do
        expect(having_one_detail(validate: true)).
          not_to have_one(:detail).validate(false)
      end

      it 'assumes validate() means validate(true)' do
        expect(having_one_detail(validate: false)).
          not_to have_one(:detail).validate
      end

      it 'matches validate(false) to having no validate option specified' do
        expect(having_one_detail).to have_one(:detail).validate(false)
      end
    end

    context 'given an association with a matching :required option' do
      it 'passes' do
        expect(having_one_detail(required: true)).
          to have_one(:detail).required
      end
    end

    context 'given an association with a non-matching :required option' do
      it 'fails with an appropriate message' do
        assertion = lambda do
          expect(having_one_detail(required: false)).
            to have_one(:detail).required
        end

        message = format_message(<<-MESSAGE, one_line: true)
          Expected Person to have a has_one association called detail (and for
          the record to fail validation if :detail is unset; i.e., either the
          association should have been defined with `required: true`, or there
          should be a presence validation on :detail)
        MESSAGE

        expect(&assertion).to fail_with_message(message)
      end
    end

    def having_one_detail(options = {})
      define_model :detail, person_id: :integer
      define_model(:person).tap do |model|
        if options.key?(:order)
          order = options.delete(:order)
          define_association_with_order(model, :has_one, :detail, order, options)
        else
          model.has_one :detail, **options
        end
      end.new
    end

    def having_one_non_existent(model_name, assoc_name, options = {})
      define_model model_name do
        has_one assoc_name, **options
      end.new
    end
  end

  context 'have_and_belong_to_many' do
    it 'accepts a valid association' do
      expect(having_and_belonging_to_many_relatives).
        to have_and_belong_to_many(:relatives)
    end

    it 'rejects a nonexistent association' do
      define_model :relative
      define_model :person
      define_model :people_relative, id: false, person_id: :integer,
        relative_id: :integer

      expect(Person.new).not_to have_and_belong_to_many(:relatives)
    end

    it 'rejects an association with a nonexistent join table' do
      define_model :relative
      define_model :person do
        has_and_belongs_to_many :relatives
      end

      expected_failure_message = "join table people_relatives doesn't exist"

      expect do
        expect(Person.new).to have_and_belong_to_many(:relatives)
      end.to fail_with_message_including(expected_failure_message)
    end

    it 'rejects an association with a join table with incorrect columns' do
      define_model :relative
      define_model :person do
        has_and_belongs_to_many :relatives
      end

      define_model :people_relative, id: false, some_crazy_id: :integer

      expect do
        expect(Person.new).to have_and_belong_to_many(:relatives)
      end.to fail_with_message_including('missing columns: person_id, relative_id')
    end

    context 'when qualified with join_table' do
      context 'and it is a symbol' do
        context 'and the association has been declared with a :join_table option' do
          context 'which is the same as the matcher' do
            context 'and the join table exists' do
              context 'and the join table has the appropriate foreign key columns' do
                it 'matches' do
                  define_model :relative

                  define_model :person do
                    has_and_belongs_to_many(
                      :relatives,
                      join_table: :people_and_their_families,
                    )
                  end

                  create_table(:people_and_their_families, id: false) do |t|
                    t.references :person
                    t.references :relative
                  end

                  build_matcher = -> do
                    have_and_belong_to_many(:relatives).
                      join_table(:people_and_their_families)
                  end

                  expect(&build_matcher).
                    to match_against(Person.new).
                    or_fail_with(<<-MESSAGE)
Did not expect Person to have a has_and_belongs_to_many association called relatives
                  MESSAGE
                end
              end

              context 'and the join table is missing columns' do
                it 'does not match, producing an appropriate failure message' do
                  define_model :relative

                  define_model :person do
                    has_and_belongs_to_many(
                      :relatives,
                      join_table: :people_and_their_families,
                    )
                  end

                  create_table(:people_and_their_families)

                  build_matcher = -> do
                    have_and_belong_to_many(:relatives).
                      join_table(:people_and_their_families)
                  end

                  expect(&build_matcher).
                    not_to match_against(Person.new).
                    and_fail_with(<<-MESSAGE)
Expected Person to have a has_and_belongs_to_many association called relatives (join table people_and_their_families missing columns: person_id, relative_id)
                  MESSAGE
                end
              end
            end

            context 'and the join table does not exist' do
              it 'does not match, producing an appropriate failure message' do
                define_model :relative

                define_model :person do
                  has_and_belongs_to_many(
                    :relatives,
                    join_table: :people_and_their_families,
                  )
                end

                build_matcher = -> do
                  have_and_belong_to_many(:relatives).
                    join_table(:family_tree)
                end

                expect(&build_matcher).
                  not_to match_against(Person.new).
                  and_fail_with(<<-MESSAGE)
Expected Person to have a has_and_belongs_to_many association called relatives (relatives should use :family_tree for :join_table option)
                MESSAGE
              end
            end
          end

          context 'which is the not the same as the matcher' do
            it 'does not match, producing an appropriate failure message' do
              define_model :relative

              define_model :person do
                has_and_belongs_to_many(
                  :relatives,
                  join_table: :people_and_their_families,
                )
              end

              create_table(:people_and_their_families, id: false) do |t|
                t.references :person
                t.references :relative
              end

              build_matcher = -> do
                have_and_belong_to_many(:relatives).join_table(:family_tree)
              end

              expect(&build_matcher).
                not_to match_against(Person.new).
                and_fail_with(<<-MESSAGE)
Expected Person to have a has_and_belongs_to_many association called relatives (relatives should use :family_tree for :join_table option)
                MESSAGE
            end
          end
        end

        context 'and the association has not been declared with a :join_table option' do
          it 'does not match, producing an appropriate failure message' do
            define_model :relative

            define_model :person do
              has_and_belongs_to_many(:relatives)
            end

            create_table(:people_relatives, id: false) do |t|
              t.references :person
              t.references :relative
            end

            build_matcher = -> do
              have_and_belong_to_many(:relatives).join_table(:family_tree)
            end

            expect(&build_matcher).
              not_to match_against(Person.new).
              and_fail_with(<<-MESSAGE)
Expected Person to have a has_and_belongs_to_many association called relatives (relatives should use :family_tree for :join_table option)
              MESSAGE
          end
        end
      end

      context 'and it is a string' do
        context 'and the association has been declared with a :join_table option' do
          context 'which is the same as the matcher' do
            context 'and the join table exists' do
              context 'and the join table has the appropriate foreign key columns' do
                it 'matches' do
                  define_model :relative

                  define_model :person do
                    has_and_belongs_to_many(
                      :relatives,
                      join_table: 'people_and_their_families',
                    )
                  end

                  create_table(:people_and_their_families, id: false) do |t|
                    t.references :person
                    t.references :relative
                  end

                  build_matcher = -> do
                    have_and_belong_to_many(:relatives).
                      join_table('people_and_their_families')
                  end

                  expect(&build_matcher).
                    to match_against(Person.new).
                    or_fail_with(<<-MESSAGE)
Did not expect Person to have a has_and_belongs_to_many association called relatives
                  MESSAGE
                end
              end

              context 'and the join table is missing columns' do
                it 'does not match, producing an appropriate failure message' do
                  define_model :relative

                  define_model :person do
                    has_and_belongs_to_many(
                      :relatives,
                      join_table: 'people_and_their_families',
                    )
                  end

                  create_table(:people_and_their_families)

                  build_matcher = -> do
                    have_and_belong_to_many(:relatives).
                      join_table('people_and_their_families')
                  end

                  expect(&build_matcher).
                    not_to match_against(Person.new).
                    and_fail_with(<<-MESSAGE)
Expected Person to have a has_and_belongs_to_many association called relatives (join table people_and_their_families missing columns: person_id, relative_id)
                  MESSAGE
                end
              end
            end

            context 'and the join table does not exist' do
              it 'does not match, producing an appropriate failure message' do
                define_model :relative

                define_model :person do
                  has_and_belongs_to_many(
                    :relatives,
                    join_table: 'people_and_their_families',
                  )
                end

                build_matcher = -> do
                  have_and_belong_to_many(:relatives).
                    join_table('family_tree')
                end

                expect(&build_matcher).
                  not_to match_against(Person.new).
                  and_fail_with(<<-MESSAGE)
Expected Person to have a has_and_belongs_to_many association called relatives (relatives should use "family_tree" for :join_table option)
                MESSAGE
              end
            end
          end

          context 'which is the not the same as the matcher' do
            it 'does not match, producing an appropriate failure message' do
              define_model :relative

              define_model :person do
                has_and_belongs_to_many(
                  :relatives,
                  join_table: 'people_and_their_families',
                )
              end

              create_table(:people_and_their_families, id: false) do |t|
                t.references :person
                t.references :relative
              end

              build_matcher = -> do
                have_and_belong_to_many(:relatives).join_table('family_tree')
              end

              expect(&build_matcher).
                not_to match_against(Person.new).
                and_fail_with(<<-MESSAGE)
Expected Person to have a has_and_belongs_to_many association called relatives (relatives should use "family_tree" for :join_table option)
                MESSAGE
            end
          end
        end

        context 'and the association has not been declared with a :join_table option' do
          it 'does not match, producing an appropriate failure message' do
            define_model :relative

            define_model :person do
              has_and_belongs_to_many(:relatives)
            end

            create_table(:people_relatives, id: false) do |t|
              t.references :person
              t.references :relative
            end

            build_matcher = -> do
              have_and_belong_to_many(:relatives).join_table('family_tree')
            end

            expect(&build_matcher).
              not_to match_against(Person.new).
              and_fail_with(<<-MESSAGE)
Expected Person to have a has_and_belongs_to_many association called relatives (relatives should use "family_tree" for :join_table option)
              MESSAGE
          end
        end
      end
    end

    context 'when the matcher is not qualified with join_table but the association has still been declared with a :join_table option' do
      context 'and the join table exists' do
        context 'and the join table has the appropriate foreign key columns' do
          it 'matches' do
            define_model :relative

            define_model :person do
              has_and_belongs_to_many(
                :relatives,
                join_table: :people_and_their_families,
              )
            end

            create_table(:people_and_their_families, id: false) do |t|
              t.references :person
              t.references :relative
            end

            build_matcher = -> { have_and_belong_to_many(:relatives) }

            expect(&build_matcher).
              to match_against(Person.new).
              or_fail_with(<<-MESSAGE)
Did not expect Person to have a has_and_belongs_to_many association called relatives
            MESSAGE
          end
        end

        context 'and the join table is missing columns' do
          it 'does not match, producing an appropriate failure message' do
            define_model :relative

            define_model :person do
              has_and_belongs_to_many(
                :relatives,
                join_table: :people_and_their_families,
              )
            end

            create_table(:people_and_their_families)

            build_matcher = -> { have_and_belong_to_many(:relatives) }

            expect(&build_matcher).
              not_to match_against(Person.new).
              and_fail_with(<<-MESSAGE)
Expected Person to have a has_and_belongs_to_many association called relatives (join table people_and_their_families missing columns: person_id, relative_id)
            MESSAGE
          end
        end
      end

      context 'and the join table does not exist' do
        it 'does not match, producing an appropriate failure message' do
          define_model :relative

          define_model :person do
            has_and_belongs_to_many(
              :relatives,
              join_table: :people_and_their_families,
            )
          end

          build_matcher = -> { have_and_belong_to_many(:relatives) }

          expect(&build_matcher).
            not_to match_against(Person.new).
            and_fail_with(<<-MESSAGE)
Expected Person to have a has_and_belongs_to_many association called relatives (join table people_and_their_families doesn't exist)
          MESSAGE
        end
      end
    end

    context 'using a custom foreign key' do
      it 'rejects an association with a join table with incorrect columns' do
        define_model :relative
        define_model :person do
          has_and_belongs_to_many :relatives,
            foreign_key: :custom_foreign_key_id
        end

        define_model :people_relative,
          id: false,
          custom_foreign_key_id: :integer,
          some_crazy_id: :integer

        expect do
          expect(Person.new).to have_and_belong_to_many(:relatives)
        end.to fail_with_message_including('missing column: relative_id')
      end
    end

    context 'using a custom association foreign key' do
      it 'rejects an association with a join table with incorrect columns' do
        define_model :relative
        define_model :person do
          has_and_belongs_to_many :relatives,
            association_foreign_key: :custom_association_foreign_key_id
        end

        define_model :people_relative,
          id: false,
          custom_association_foreign_key_id: :integer,
          some_crazy_id: :integer

        expect do
          expect(Person.new).to have_and_belong_to_many(:relatives)
        end.to fail_with_message_including('missing column: person_id')
      end

      it 'accepts foreign keys when they are symbols' do
        define_model :relative
        define_model :person do
          has_and_belongs_to_many :relatives,
            foreign_key: :some_foreign_key_id,
            association_foreign_key: :custom_association_foreign_key_id
        end

        define_model :people_relative,
          id: false,
          custom_association_foreign_key_id: :integer,
          some_foreign_key_id: :integer

        expect(Person.new).to have_and_belong_to_many(:relatives)
      end
    end

    it 'rejects an association of the wrong type' do
      define_model :relative, person_id: :integer
      define_model :person do
        has_many :relatives
      end

      expect(Person.new).not_to have_and_belong_to_many(:relatives)
    end

    it 'accepts an association with a valid :conditions option' do
      define_model :relative, adopted: :boolean
      define_model(:person).tap do |model|
        define_association_with_conditions(model, :has_and_belongs_to_many, :relatives, adopted: true)
      end
      define_model :people_relative, id: false, person_id: :integer,
        relative_id: :integer

      expect(Person.new).to have_and_belong_to_many(:relatives).conditions(adopted: true)
    end

    it 'rejects an association with a bad :conditions option' do
      define_model :relative, adopted: :boolean
      define_model :person do
        has_and_belongs_to_many :relatives
      end
      define_model :people_relative, id: false, person_id: :integer,
        relative_id: :integer

      expect(Person.new).not_to have_and_belong_to_many(:relatives).conditions(adopted: true)
    end

    it 'accepts an association without a :class_name option' do
      expect(having_and_belonging_to_many_relatives).
        to have_and_belong_to_many(:relatives).class_name('Relative')
    end

    it 'accepts an association with a valid :class_name option' do
      define_model :person_relative, adopted: :boolean
      define_model :person do
        has_and_belongs_to_many :relatives, class_name: 'PersonRelative'
      end

      define_model :people_person_relative, person_id: :integer,
        person_relative_id: :integer

      expect(Person.new).to have_and_belong_to_many(:relatives).class_name('PersonRelative')
    end

    it 'rejects an association with a bad :class_name option' do
      expect(having_and_belonging_to_many_relatives).
        not_to have_and_belong_to_many(:relatives).class_name('PersonRelatives')
    end

    it 'rejects an association with non-existent implicit class name' do
      expect(having_and_belonging_to_many_non_existent_class(:person, :relatives)).
        not_to have_and_belong_to_many(:relatives)
    end

    it 'rejects an association with non-existent explicit class name' do
      expect(having_and_belonging_to_many_non_existent_class(:person, :relatives, class_name: 'Relative')).
        not_to have_and_belong_to_many(:relatives)
    end

    it 'adds error message when rejecting an association with non-existent class' do
      message = 'Expected Person to have a has_and_belongs_to_many association called relatives (Relative2 does not exist)'
      expect {
        expect(having_and_belonging_to_many_non_existent_class(:person, :relatives, class_name: 'Relative2')).
          to have_and_belong_to_many(:relatives)
      }.to fail_with_message(message)
    end

    it 'accepts an association with a namespaced class name' do
      possible_join_table_names = [:groups_users, :models_groups_users, :groups_models_users]
      possible_join_table_names.each do |join_table_name|
        create_table join_table_name, id: false do |t|
          t.integer :group_id
          t.integer :user_id
        end
      end
      define_module 'Models'
      define_model 'Models::Group'
      user_model = define_model 'Models::User' do
        has_and_belongs_to_many :groups, class_name: 'Group'
      end

      expect(user_model.new).
        to have_and_belong_to_many(:groups).
        class_name('Group')
    end

    it 'resolves class_name within the context of the namespace before the global namespace' do
      possible_join_table_names = [:groups_users, :models_groups_users, :groups_models_users]
      possible_join_table_names.each do |join_table_name|
        create_table join_table_name, id: false do |t|
          t.integer :group_id
          t.integer :user_id
        end
      end
      define_module 'Models'
      define_model 'Group'
      define_model 'Models::Group'
      user_model = define_model 'Models::User' do
        has_and_belongs_to_many :groups, class_name: 'Group'
      end

      expect(user_model.new).
        to have_and_belong_to_many(:groups).
        class_name('Group')
    end

    it 'accepts an association with a matching :autosave option' do
      define_model :relatives, adopted: :boolean
      define_model :person do
        has_and_belongs_to_many :relatives, autosave: true
      end
      define_model :people_relative, person_id: :integer,
        relative_id: :integer
      expect(Person.new).to have_and_belong_to_many(:relatives).autosave(true)
    end

    it 'rejects an association with a non-matching :autosave option with the correct message' do
      define_model :relatives, adopted: :boolean
      define_model :person do
        has_and_belongs_to_many :relatives
      end
      define_model :people_relative, person_id: :integer,
        relative_id: :integer

      message = 'Expected Person to have a has_and_belongs_to_many association called relatives (relatives should have autosave set to true)'
      expect {
        expect(Person.new).to have_and_belong_to_many(:relatives).autosave(true)
      }.to fail_with_message(message)
    end

    context 'validate' do
      it 'accepts when the :validate option matches' do
        expect(having_and_belonging_to_many_relatives(validate: false)).
          to have_and_belong_to_many(:relatives).validate(false)
      end

      it 'rejects when the :validate option does not match' do
        expect(having_and_belonging_to_many_relatives(validate: true)).
          to have_and_belong_to_many(:relatives).validate(false)
      end

      it 'assumes validate() means validate(true)' do
        expect(having_and_belonging_to_many_relatives(validate: false)).
          not_to have_and_belong_to_many(:relatives).validate
      end

      it 'matches validate(false) to having no validate option specified' do
        expect(having_and_belonging_to_many_relatives).
          to have_and_belong_to_many(:relatives).validate(false)
      end
    end

    if rails_version >= 8.1
      context 'deprecated' do
        it 'accepts when the :deprecated option matches' do
          expect(having_and_belonging_to_many_relatives(deprecated: false)).
            to have_and_belong_to_many(:relatives).deprecated(false)
        end

        it 'rejects when the :deprecated option does not match' do
          expect(having_and_belonging_to_many_relatives(deprecated: true)).
            to have_and_belong_to_many(:relatives).deprecated(false)
        end

        it 'assumes deprecated() means deprecated(true)' do
          expect(having_and_belonging_to_many_relatives(deprecated: false)).
            not_to have_and_belong_to_many(:relatives).deprecated
        end

        it 'matches deprecated(false) to having no deprecated option specified' do
          expect(having_and_belonging_to_many_relatives).
            to have_and_belong_to_many(:relatives).deprecated(false)
        end

        it 'rejects an association with a non-matching :deprecated option with the correct message' do
          define_model :relatives, adopted: :boolean
          define_model :person do
            has_and_belongs_to_many :relatives
          end
          define_model :people_relative, person_id: :integer,
            relative_id: :integer

          message = 'Expected Person to have a has_and_belongs_to_many association called relatives (relatives should have deprecated set to true)'
          expect do
            expect(Person.new).to have_and_belong_to_many(:relatives).deprecated(true)
          end.to fail_with_message(message)
        end
      end
    end

    def having_and_belonging_to_many_relatives(_options = {})
      define_model :relative
      define_model :people_relative, id: false, person_id: :integer,
        relative_id: :integer
      define_model :person do
        has_and_belongs_to_many :relatives
      end.new
    end

    def having_and_belonging_to_many_non_existent_class(model_name, assoc_name, options = {})
      define_model model_name do
        has_and_belongs_to_many assoc_name, **options
      end.new
    end
  end

  context 'delegated_types' do
    it 'accepts a good association with the default foreign key' do
      expect(delegating_type_to_drivable).to have_delegated_type(:drivable)
    end

    it 'rejects a nonexistent association' do
      expect(define_model(:vehicle).new).not_to have_delegated_type(:drivable)
    end

    it 'accepts an association specifying the types' do
      expect(delegating_type_to_drivable).to have_delegated_type(:drivable).types(['Car'])
    end

    it 'rejects an association one wrong type' do
      expect(delegating_type_to_drivable).not_to have_delegated_type(:drivable).types(['Car', 'Truck'])
    end

    it 'rejects an association with all wrong types' do
      expect(delegating_type_to_drivable).not_to have_delegated_type(:drivable).types(['Truck'])
    end

    it 'rejects an association that has a nonexistent foreign key' do
      define_model :car
      expect((define_model(:vehicle) { delegated_type :drivable, types: ['Car'] }).new).not_to have_delegated_type(:drivable)
    end

    it 'accepts an association with an existing custom foreign key' do
      expect(delegating_type_to_drivable(foreign_key: 'drivable_uuid')).to have_delegated_type(:drivable)
    end

    it 'accepts an association using an existing custom primary key' do
      define_model :car, custom_primary_key: :integer
      define_model :vehicle, drivable_id: :integer, drivable_type: :string do
        delegated_type :drivable, types: ['Car'], primary_key: 'custom_primary_key'
      end

      expect(Vehicle.new).to have_delegated_type(:drivable)
    end

    it 'rejects an association with a bad :primary_key option' do
      matcher = have_delegated_type(:drivable).with_primary_key(:custom_primary_key)

      expect(delegating_type_to_drivable).not_to matcher

      expect(matcher.failure_message).to match(/Vehicle does not have a custom_primary_key primary key/)
    end

    it 'accepts an association with a valid :dependent option' do
      expect(delegating_type_to_drivable(dependent: :destroy)).
        to have_delegated_type(:drivable).dependent(:destroy)
    end

    it 'rejects an association with a bad :dependent option' do
      expect(delegating_type_to_drivable).not_to have_delegated_type(:drivable).dependent(:destroy)
    end

    it 'accepts an association with a valid :counter_cache option' do
      expect(delegating_type_to_drivable(counter_cache: :attribute_count)).
        to have_delegated_type(:drivable).counter_cache(:attribute_count)
    end

    it 'defaults :counter_cache to true' do
      expect(delegating_type_to_drivable(counter_cache: true)).
        to have_delegated_type(:drivable).counter_cache
    end

    it 'rejects an association with a bad :counter_cache option' do
      expect(delegating_type_to_drivable(counter_cache: :attribute_count)).
        not_to have_delegated_type(:drivable).counter_cache(true)
    end

    it 'rejects an association that has no :counter_cache option' do
      expect(delegating_type_to_drivable).not_to have_delegated_type(:drivable).counter_cache
    end

    it 'accepts an association with a valid :inverse_of option' do
      define_model :vehicle, drivable_id: :integer, drivable_type: :string do
        delegated_type :drivable, types: ['Car'], inverse_of: :vehicle
      end

      define_model :car do
        has_one :vehicle, as: :drivable, inverse_of: :drivable
      end

      expect(Vehicle.new).to have_delegated_type(:drivable).inverse_of(:vehicle)
    end

    it 'rejects an association with a bad :inverse_of option' do
      define_model :vehicle, drivable_id: :integer, drivable_type: :string do
        delegated_type :drivable, types: ['Car'], inverse_of: :vehicle
      end

      define_model :car do
        has_one :vehicle, as: :drivable, inverse_of: :drivable
      end

      expect(Vehicle.new).not_to have_delegated_type(:drivable).inverse_of(:other_vehicle)
    end

    it 'rejects an association that has no :inverse_of option' do
      expect(delegating_type_to_drivable).not_to have_delegated_type(:drivable).inverse_of(:vehicle)
    end

    it 'accepts an association with a valid :conditions option' do
      define_model :vehicle, drivable_id: :integer, drivable_type: :string do
        delegated_type :drivable, types: ['Car'], scope: -> { where(color: 'red') }
      end

      expect(Vehicle.new).to have_delegated_type(:drivable).conditions(color: 'red')
    end

    it 'rejects an association with a bad :conditions option' do
      define_model :vehicle, drivable_id: :integer, drivable_type: :string do
        delegated_type :drivable, types: ['Car'], scope: -> { where(color: 'red') }
      end

      expect(Vehicle.new).not_to have_delegated_type(:drivable).conditions(color: 'blue')
    end

    it 'accepts an association with a matching :autosave option' do
      expect(delegating_type_to_drivable(autosave: true)).to have_delegated_type(:drivable).autosave(true)
    end

    it 'rejects an association with a non-matching :autosave option with the correct message' do
      message = 'Expected Vehicle to have a belongs_to association called drivable (drivable should have autosave set to true)'

      expect {
        expect(delegating_type_to_drivable(autosave: false)).to have_delegated_type(:drivable).autosave(true)
      }.to fail_with_message(message)
    end

    context 'an association with a :validate option' do
      [false, true].each do |validate_value|
        context "when the model has validate: #{validate_value}" do
          it 'accepts a matching validate option' do
            expect(delegating_type_to_drivable(validate: validate_value)).
              to have_delegated_type(:drivable).validate(validate_value)
          end

          it 'rejects a non-matching validate option' do
            expect(delegating_type_to_drivable(validate: validate_value)).
              not_to have_delegated_type(:drivable).validate(!validate_value)
          end

          it 'defaults to validate(true)' do
            if validate_value
              expect(delegating_type_to_drivable(validate: validate_value)).
                to have_delegated_type(:drivable).validate
            else
              expect(delegating_type_to_drivable(validate: validate_value)).
                not_to have_delegated_type(:drivable).validate
            end
          end

          it 'will not break matcher when validate option is unspecified' do
            expect(delegating_type_to_drivable(validate: validate_value)).to have_delegated_type(:drivable)
          end
        end
      end
    end

    context 'an association without a :validate option' do
      it 'accepts validate(false)' do
        expect(delegating_type_to_drivable).to have_delegated_type(:drivable).validate(false)
      end

      it 'rejects validate(true)' do
        expect(delegating_type_to_drivable).not_to have_delegated_type(:drivable).validate(true)
      end

      it 'rejects validate()' do
        expect(delegating_type_to_drivable).not_to have_delegated_type(:drivable).validate
      end
    end

    context 'an association with a :touch option' do
      [false, true].each do |touch_value|
        context "when the model has touch: #{touch_value}" do
          it 'accepts a matching touch option' do
            expect(delegating_type_to_drivable(touch: touch_value)).
              to have_delegated_type(:drivable).touch(touch_value)
          end

          it 'rejects a non-matching touch option' do
            expect(delegating_type_to_drivable(touch: touch_value)).
              not_to have_delegated_type(:drivable).touch(!touch_value)
          end

          it 'defaults to touch(true)' do
            if touch_value
              expect(delegating_type_to_drivable(touch: touch_value)).
                to have_delegated_type(:drivable).touch
            else
              expect(delegating_type_to_drivable(touch: touch_value)).
                not_to have_delegated_type(:drivable).touch
            end
          end

          it 'will not break matcher when touch option is unspecified' do
            expect(delegating_type_to_drivable(touch: touch_value)).to have_delegated_type(:drivable)
          end
        end
      end
    end

    context 'an association without a :touch option' do
      it 'accepts touch(false)' do
        expect(delegating_type_to_drivable).to have_delegated_type(:drivable).touch(false)
      end

      it 'rejects touch(true)' do
        expect(delegating_type_to_drivable).not_to have_delegated_type(:drivable).touch(true)
      end

      it 'rejects touch()' do
        expect(delegating_type_to_drivable).not_to have_delegated_type(:drivable).touch
      end
    end

    if rails_version >= 8.1
      context 'an association with a :deprecated option' do
        [false, true].each do |deprecated_value|
          context "when the model has deprecated: #{deprecated_value}" do
            it 'accepts a matching deprecated option' do
              expect(delegating_type_to_drivable(deprecated: deprecated_value)).
                to have_delegated_type(:drivable).deprecated(deprecated_value)
            end

            it 'rejects a non-matching deprecated option' do
              expect(delegating_type_to_drivable(deprecated: deprecated_value)).
                not_to have_delegated_type(:drivable).deprecated(!deprecated_value)
            end

            it 'defaults to deprecated(true)' do
              if deprecated_value
                expect(delegating_type_to_drivable(deprecated: deprecated_value)).
                  to have_delegated_type(:drivable).deprecated
              else
                expect(delegating_type_to_drivable(deprecated: deprecated_value)).
                  not_to have_delegated_type(:drivable).deprecated
              end
            end

            it 'will not break matcher when deprecated option is unspecified' do
              expect(delegating_type_to_drivable(deprecated: deprecated_value)).to have_delegated_type(:drivable)
            end
          end
        end
      end

      context 'an association without a :deprecated option' do
        it 'accepts deprecated(false)' do
          expect(delegating_type_to_drivable).to have_delegated_type(:drivable).deprecated(false)
        end

        it 'rejects deprecated(true)' do
          expect(delegating_type_to_drivable).not_to have_delegated_type(:drivable).deprecated(true)
        end

        it 'rejects deprecated()' do
          expect(delegating_type_to_drivable).not_to have_delegated_type(:drivable).deprecated
        end
      end

      it 'rejects an association with a non-matching :deprecated option with the correct message' do
        message = 'Expected Vehicle to have a belongs_to association called drivable (drivable should have deprecated set to true)'

        expect do
          expect(delegating_type_to_drivable(deprecated: false)).to have_delegated_type(:drivable).deprecated(true)
        end.to fail_with_message(message)
      end
    end

    context 'given the association is neither configured to be required nor optional' do
      context 'when qualified with required(true)' do
        context 'when belongs_to is configured to be required by default' do
          it 'passes' do
            with_belongs_to_as_required_by_default do
              expect(delegating_type_to_drivable).to have_delegated_type(:drivable).required(true)
            end
          end
        end

        context 'when belongs_to is not configured to be required by default' do
          it 'fails with an appropriate message' do
            with_belongs_to_as_optional_by_default do
              assertion = lambda do
                expect(delegating_type_to_drivable).to have_delegated_type(:drivable).required(true)
              end

              message = format_message(<<-MESSAGE, one_line: true)
                Expected Vehicle to have a belongs_to association called drivable
                (and for the record to fail validation if :drivable is unset;
                i.e., either the association should have been defined with
                `required: true`, or there should be a presence validation on
                :drivable)
              MESSAGE

              expect(&assertion).to fail_with_message(message)
            end
          end
        end
      end

      context 'when qualified with required(false)' do
        context 'when belongs_to is configured to be required by default' do
          it 'fails with an appropriate message' do
            with_belongs_to_as_required_by_default do
              assertion = lambda do
                expect(delegating_type_to_drivable).to have_delegated_type(:drivable).required(false)
              end

              message = format_message(<<-MESSAGE, one_line: true)
                Expected Vehicle to have a belongs_to association called drivable
                (and for the record not to fail validation if :drivable is
                unset; i.e., either the association should have been defined
                with `required: false`, or there should not be a presence
                validation on :drivable)
              MESSAGE

              expect(&assertion).to fail_with_message(message)
            end
          end
        end

        context 'when belongs_to is not configured to be required by default' do
          it 'passes' do
            with_belongs_to_as_optional_by_default do
              expect(delegating_type_to_drivable).to have_delegated_type(:drivable).required(false)
            end
          end
        end
      end

      context 'when qualified with optional(true)' do
        context 'when belongs_to is configured to be required by default' do
          it 'fails with an appropriate message' do
            with_belongs_to_as_required_by_default do
              assertion = lambda do
                expect(delegating_type_to_drivable).to have_delegated_type(:drivable).optional(true)
              end

              message = format_message(<<-MESSAGE, one_line: true)
                Expected Vehicle to have a belongs_to association called drivable
                (and for the record not to fail validation if :drivable is
                unset; i.e., either the association should have been defined
                with `optional: true`, or there should not be a presence
                validation on :drivable)
              MESSAGE

              expect(&assertion).to fail_with_message(message)
            end
          end
        end

        context 'when belongs_to is not configured to be required by default' do
          it 'passes' do
            with_belongs_to_as_optional_by_default do
              expect(delegating_type_to_drivable).to have_delegated_type(:drivable).optional(true)
            end
          end
        end
      end

      context 'when qualified with optional(false)' do
        context 'when belongs_to is configured to be required by default' do
          it 'passes' do
            with_belongs_to_as_required_by_default do
              expect(delegating_type_to_drivable).to have_delegated_type(:drivable).optional(false)
            end
          end
        end

        context 'when belongs_to is not configured to be required by default' do
          it 'fails with an appropriate message' do
            with_belongs_to_as_optional_by_default do
              assertion = lambda do
                expect(delegating_type_to_drivable).to have_delegated_type(:drivable).optional(false)
              end

              message = format_message(<<-MESSAGE, one_line: true)
                Expected Vehicle to have a belongs_to association called drivable
                (and for the record to fail validation if :drivable is
                unset; i.e., either the association should have been defined
                with `optional: false`, or there should be a presence
                validation on :drivable)
              MESSAGE

              expect(&assertion).to fail_with_message(message)
            end
          end
        end
      end

      context 'when qualified with nothing' do
        context 'when belongs_to is configured to be required by default' do
          it 'passes' do
            with_belongs_to_as_required_by_default do
              expect(delegating_type_to_drivable).to have_delegated_type(:drivable)
            end
          end
        end

        context 'when belongs_to is not configured to be required by default' do
          it 'passes' do
            with_belongs_to_as_optional_by_default do
              expect(delegating_type_to_drivable).to have_delegated_type(:drivable)
            end
          end

          context 'and a presence validation is on the attribute instead of using required: true' do
            it 'passes' do
              with_belongs_to_as_optional_by_default do
                record = delegating_type_to_drivable do
                  validates_presence_of :drivable
                end

                expect(record).to have_delegated_type(:drivable)
              end
            end
          end

          context 'and a presence validation is on the attribute with a condition' do
            context 'and the condition is true' do
              it 'passes' do
                with_belongs_to_as_optional_by_default do
                  record = delegating_type_to_drivable do
                    attr_accessor :condition

                    validates_presence_of :drivable, if: :condition
                  end

                  record.condition = true

                  expect(record).to have_delegated_type(:drivable)
                end
              end
            end

            context 'and the condition is false' do
              it 'passes' do
                with_belongs_to_as_optional_by_default do
                  record = delegating_type_to_drivable do
                    attr_accessor :condition

                    validates_presence_of :drivable, if: :condition
                  end

                  record.condition = false

                  expect(record).to have_delegated_type(:drivable)
                end
              end
            end
          end
        end
      end
    end

    context 'given the association is configured with required: true' do
      context 'when qualified with required(true)' do
        it 'passes' do
          expect(delegating_type_to_drivable(required: true)).
            to have_delegated_type(:drivable).required(true)
        end
      end

      context 'when qualified with required(false)' do
        it 'passes' do
          assertion = lambda do
            expect(delegating_type_to_drivable(required: true)).
              to have_delegated_type(:drivable).required(false)
          end

          message = format_message(<<-MESSAGE, one_line: true)
            Expected Vehicle to have a belongs_to association called drivable (and
            for the record not to fail validation if :drivable is unset; i.e.,
            either the association should have been defined with `required:
            false`, or there should not be a presence validation on :drivable)
          MESSAGE

          expect(&assertion).to fail_with_message(message)
        end
      end

      context 'when qualified with optional(true)' do
        it 'fails with an appropriate message' do
          assertion = lambda do
            expect(delegating_type_to_drivable(required: true)).
              to have_delegated_type(:drivable).optional(true)
          end

          message = format_message(<<-MESSAGE, one_line: true)
            Expected Vehicle to have a belongs_to association called drivable
            (and for the record not to fail validation if :drivable is unset;
            i.e., either the association should have been defined with
            `optional: true`, or there should not be a presence validation on
            :drivable)
          MESSAGE

          expect(&assertion).to fail_with_message(message)
        end
      end

      context 'when qualified with optional(false)' do
        it 'passes' do
          expect(delegating_type_to_drivable(required: true)).
            to have_delegated_type(:drivable).optional(false)
        end
      end

      context 'when qualified with nothing' do
        it 'passes' do
          expect(delegating_type_to_drivable(required: true)).to have_delegated_type(:drivable)
        end
      end
    end

    context 'given the association is configured as optional: true' do
      context 'when qualified with required(true)' do
        it 'fails with an appropriate message' do
          assertion = lambda do
            expect(delegating_type_to_drivable(optional: true)).
              to have_delegated_type(:drivable).required(true)
          end

          message = format_message(<<-MESSAGE, one_line: true)
            Expected Vehicle to have a belongs_to association called drivable
            (and for the record to fail validation if :drivable is unset; i.e.,
            either the association should have been defined with `required:
            true`, or there should be a presence validation on :drivable)
          MESSAGE

          expect(&assertion).to fail_with_message(message)
        end
      end

      context 'when qualified with required(false)' do
        it 'passes' do
          expect(delegating_type_to_drivable(optional: true)).
            to have_delegated_type(:drivable).required(false)
        end
      end

      context 'when qualified with optional(true)' do
        it 'passes' do
          expect(delegating_type_to_drivable(optional: true)).
            to have_delegated_type(:drivable).optional(true)
        end
      end

      context 'when qualified with optional(false)' do
        it 'fails with an appropriate message' do
          assertion = lambda do
            expect(delegating_type_to_drivable(optional: true)).
              to have_delegated_type(:drivable).optional(false)
          end

          message = format_message(<<-MESSAGE, one_line: true)
            Expected Vehicle to have a belongs_to association called drivable
            (and for the record to fail validation if :drivable is unset; i.e.,
            either the association should have been defined with `optional:
            false`, or there should be a presence validation on :drivable)
          MESSAGE

          expect(&assertion).to fail_with_message(message)
        end
      end

      context 'when qualified with nothing' do
        it 'fails with an appropriate message' do
          assertion = lambda do
            expect(delegating_type_to_drivable(optional: true)).
              to have_delegated_type(:drivable)
          end

          message = format_message(<<-MESSAGE, one_line: true)
            Expected Vehicle to have a belongs_to association called drivable
            (and for the record to fail validation if :drivable is unset; i.e.,
            either the association should have been defined with `required:
            true`, or there should be a presence validation on :drivable)
          MESSAGE

          expect(&assertion).to fail_with_message(message)
        end
      end
    end

    context 'when the model ensures the association is set' do
      context 'and the matcher is not qualified with anything' do
        context 'and the matcher is not qualified with without_validating_presence' do
          it 'fails with an appropriate message' do
            record = delegating_type_to_drivable do
              before_validation :ensure_drivable_is_set

              def ensure_drivable_is_set
                self.drivable = Car.create
              end
            end

            assertion = lambda do
              with_belongs_to_as_required_by_default do
                expect(record).to have_delegated_type(:drivable)
              end
            end

            message = format_message(<<-MESSAGE, one_line: true)
              Expected Vehicle to have a belongs_to association called drivable (and
              for the record to fail validation if :drivable is unset; i.e.,
              either the association should have been defined with `required:
              true`, or there should be a presence validation on :drivable)
            MESSAGE

            expect(&assertion).to fail_with_message(message)
          end
        end

        context 'and the matcher is qualified with without_validating_presence' do
          it 'passes' do
            record = delegating_type_to_drivable do
              before_validation :ensure_drivable_is_set

              def ensure_drivable_is_set
                self.drivable = Car.create
              end
            end

            with_belongs_to_as_required_by_default do
              expect(record).
                to have_delegated_type(:drivable).
                without_validating_presence
            end
          end
        end
      end

      context 'and the matcher is qualified with required' do
        context 'and the matcher is not qualified with without_validating_presence' do
          it 'fails with an appropriate message' do
            record = delegating_type_to_drivable do
              before_validation :ensure_drivable_is_set

              def ensure_drivable_is_set
                self.drivable = Car.create
              end
            end

            assertion = lambda do
              with_belongs_to_as_required_by_default do
                expect(record).to have_delegated_type(:drivable).required
              end
            end

            message = format_message(<<-MESSAGE, one_line: true)
              Expected Vehicle to have a belongs_to association called drivable
              (and for the record to fail validation if :drivable is unset; i.e.,
              either the association should have been defined with `required:
              true`, or there should be a presence validation on :drivable)
            MESSAGE

            expect(&assertion).to fail_with_message(message)
          end
        end

        context 'and the matcher is also qualified with without_validating_presence' do
          it 'passes' do
            record = delegating_type_to_drivable do
              before_validation :ensure_drivable_is_set

              def ensure_drivable_is_set
                self.drivable = Car.create
              end
            end

            with_belongs_to_as_required_by_default do
              expect(record).
                to have_delegated_type(:drivable).
                required.
                without_validating_presence
            end
          end
        end
      end
    end
  end

  def delegating_type_to_drivable(options = {}, &block)
    foreign_key = options[:foreign_key] || 'drivable_id'
    define_model :car
    define_model :vehicle, { "#{foreign_key}": :integer, drivable_type: :string } do
      delegated_type :drivable, types: ['Car'], **options
      if block
        class_eval(&block)
      end
    end.new
  end

  def define_association_with_conditions(model, macro, name, conditions, _other_options = {})
    model.__send__(macro, name, proc { where(conditions) }, **{})
  end

  def define_association_with_order(model, macro, name, order, _other_options = {})
    model.__send__(macro, name, proc { order(order) }, **{})
  end

  def dependent_options
    [:destroy, :delete, :nullify, :restrict_with_exception, :restrict_with_error]
  end
end
